4 * Purple is the legal property of its developers, whose names are too numerous
5 * to list here. Please refer to the COPYRIGHT file distributed with this
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
22 #define _PURPLE_PLUGIN_C_
26 #include "accountopt.h"
28 #include "dbus-maybe.h"
44 } PurplePluginIpcInfo
;
49 PurpleSignalMarshalFunc marshal
;
53 PurpleValue
*ret_value
;
55 } PurplePluginIpcCommand
;
57 static GList
*search_paths
= NULL
;
58 static GList
*plugins
= NULL
;
59 static GList
*loaded_plugins
= NULL
;
60 static GList
*protocol_plugins
= NULL
;
62 static GList
*load_queue
= NULL
;
63 static GList
*plugin_loaders
= NULL
;
64 static GList
*plugins_to_disable
= NULL
;
67 static void (*probe_cb
)(void *) = NULL
;
68 static void *probe_cb_data
= NULL
;
69 static void (*load_cb
)(PurplePlugin
*, void *) = NULL
;
70 static void *load_cb_data
= NULL
;
71 static void (*unload_cb
)(PurplePlugin
*, void *) = NULL
;
72 static void *unload_cb_data
= NULL
;
77 has_file_extension(const char *filename
, const char *ext
)
81 if (filename
== NULL
|| *filename
== '\0' || ext
== NULL
)
85 len
= strlen(filename
) - extlen
;
90 return (strncmp(filename
+ len
, ext
, extlen
) == 0);
94 is_native(const char *filename
)
96 const char *last_period
;
98 last_period
= strrchr(filename
, '.');
99 if (last_period
== NULL
)
102 return purple_strequal(last_period
, ".dll") ||
103 purple_strequal(last_period
, ".sl") ||
104 purple_strequal(last_period
, ".so");
108 purple_plugin_get_basename(const char *filename
)
110 const char *basename
;
111 const char *last_period
;
113 basename
= strrchr(filename
, G_DIR_SEPARATOR
);
114 if (basename
!= NULL
)
119 if (is_native(basename
) &&
120 ((last_period
= strrchr(basename
, '.')) != NULL
))
121 return g_strndup(basename
, (last_period
- basename
));
123 return g_strdup(basename
);
127 loader_supports_file(PurplePlugin
*loader
, const char *filename
)
131 for (exts
= PURPLE_PLUGIN_LOADER_INFO(loader
)->exts
; exts
!= NULL
; exts
= exts
->next
) {
132 if (has_file_extension(filename
, (char *)exts
->data
)) {
140 static PurplePlugin
*
141 find_loader_for_plugin(const PurplePlugin
*plugin
)
143 PurplePlugin
*loader
;
146 if (plugin
->path
== NULL
)
149 for (l
= purple_plugins_get_loaded(); l
!= NULL
; l
= l
->next
) {
152 if (loader
->info
->type
== PURPLE_PLUGIN_LOADER
&&
153 loader_supports_file(loader
, plugin
->path
)) {
164 #endif /* PURPLE_PLUGINS */
167 * Negative if a before b, 0 if equal, positive if a after b.
170 compare_prpl(PurplePlugin
*a
, PurplePlugin
*b
)
172 if(PURPLE_IS_PROTOCOL_PLUGIN(a
)) {
173 if(PURPLE_IS_PROTOCOL_PLUGIN(b
))
174 return strcmp(a
->info
->name
, b
->info
->name
);
178 if(PURPLE_IS_PROTOCOL_PLUGIN(b
))
186 purple_plugin_new(gboolean native
, const char *path
)
188 PurplePlugin
*plugin
;
190 plugin
= g_new0(PurplePlugin
, 1);
192 plugin
->native_plugin
= native
;
193 plugin
->path
= g_strdup(path
);
195 PURPLE_DBUS_REGISTER_POINTER(plugin
, PurplePlugin
);
201 purple_plugin_probe(const char *filename
)
203 #ifdef PURPLE_PLUGINS
204 PurplePlugin
*plugin
= NULL
;
205 PurplePlugin
*loader
;
207 gchar
*basename
= NULL
;
208 gboolean (*purple_init_plugin
)(PurplePlugin
*);
210 purple_debug_misc("plugins", "probing %s\n", filename
);
211 g_return_val_if_fail(filename
!= NULL
, NULL
);
213 if (!g_file_test(filename
, G_FILE_TEST_EXISTS
))
216 /* If this plugin has already been probed then exit */
217 basename
= purple_plugin_get_basename(filename
);
218 plugin
= purple_plugins_find_with_basename(basename
);
222 if (purple_strequal(filename
, plugin
->path
))
224 else if (!purple_plugin_is_unloadable(plugin
))
226 purple_debug_warning("plugins", "Not loading %s. "
227 "Another plugin with the same name (%s) has already been loaded.\n",
228 filename
, plugin
->path
);
233 /* The old plugin was a different file and it was unloadable.
234 * There's no guarantee that this new file with the same name
235 * will be loadable, but unless it fails in one of the silent
236 * ways and the first one didn't, it's not any worse. The user
237 * will still see a greyed-out plugin, which is what we want. */
238 purple_plugin_destroy(plugin
);
242 plugin
= purple_plugin_new(has_file_extension(filename
, G_MODULE_SUFFIX
), filename
);
244 if (plugin
->native_plugin
) {
247 /* Suppress error popups for failing to load plugins */
248 UINT old_error_mode
= SetErrorMode(SEM_FAILCRITICALERRORS
);
252 * We pass G_MODULE_BIND_LOCAL here to prevent symbols from
253 * plugins being added to the global name space.
255 * G_MODULE_BIND_LOCAL was added in glib 2.3.3.
257 plugin
->handle
= g_module_open(filename
, G_MODULE_BIND_LOCAL
);
259 if (plugin
->handle
== NULL
)
261 const char *error
= g_module_error();
262 if (error
!= NULL
&& purple_str_has_prefix(error
, filename
))
264 error
= error
+ strlen(filename
);
266 /* These are just so we don't crash. If we
267 * got this far, they should always be true. */
274 if (error
== NULL
|| !*error
)
276 plugin
->error
= g_strdup(_("Unknown error"));
277 purple_debug_error("plugins", "%s is not loadable: Unknown error\n",
282 plugin
->error
= g_strdup(error
);
283 purple_debug_error("plugins", "%s is not loadable: %s\n",
284 plugin
->path
, plugin
->error
);
286 plugin
->handle
= g_module_open(filename
, G_MODULE_BIND_LAZY
| G_MODULE_BIND_LOCAL
);
288 if (plugin
->handle
== NULL
)
291 /* Restore the original error mode */
292 SetErrorMode(old_error_mode
);
294 purple_plugin_destroy(plugin
);
299 /* We were able to load the plugin with lazy symbol binding.
300 * This means we're missing some symbol. Mark it as
301 * unloadable and keep going so we get the info to display
302 * to the user so they know to rebuild this plugin. */
303 plugin
->unloadable
= TRUE
;
307 if (!g_module_symbol(plugin
->handle
, "purple_init_plugin",
310 purple_debug_error("plugins", "%s is not usable because the "
311 "'purple_init_plugin' symbol could not be "
312 "found. Does the plugin call the "
313 "PURPLE_INIT_PLUGIN() macro?\n", plugin
->path
);
315 g_module_close(plugin
->handle
);
316 error
= g_module_error();
318 purple_debug_error("plugins", "Error closing module %s: %s\n",
319 plugin
->path
, error
);
320 plugin
->handle
= NULL
;
323 /* Restore the original error mode */
324 SetErrorMode(old_error_mode
);
326 purple_plugin_destroy(plugin
);
329 purple_init_plugin
= unpunned
;
332 /* Restore the original error mode */
333 SetErrorMode(old_error_mode
);
337 loader
= find_loader_for_plugin(plugin
);
339 if (loader
== NULL
) {
340 purple_plugin_destroy(plugin
);
344 purple_init_plugin
= PURPLE_PLUGIN_LOADER_INFO(loader
)->probe
;
347 if (!purple_init_plugin(plugin
) || plugin
->info
== NULL
)
349 purple_plugin_destroy(plugin
);
352 else if (plugin
->info
->ui_requirement
&&
353 !purple_strequal(plugin
->info
->ui_requirement
, purple_core_get_ui()))
355 plugin
->error
= g_strdup_printf(_("You are using %s, but this plugin requires %s."),
356 purple_core_get_ui(), plugin
->info
->ui_requirement
);
357 purple_debug_error("plugins", "%s is not loadable: The UI requirement is not met. (%s)\n", plugin
->path
, plugin
->error
);
358 plugin
->unloadable
= TRUE
;
363 * Check to make sure a plugin has defined an id.
364 * Not having this check caused purple_plugin_unload to
365 * enter an infinite loop in certain situations by passing
366 * purple_find_plugin_by_id a NULL value. -- ecoffey
368 if (plugin
->info
->id
== NULL
|| *plugin
->info
->id
== '\0')
370 plugin
->error
= g_strdup(_("This plugin has not defined an ID."));
371 purple_debug_error("plugins", "%s is not loadable: info->id is not defined.\n", plugin
->path
);
372 plugin
->unloadable
= TRUE
;
376 /* Really old plugins. */
377 if (plugin
->info
->magic
!= PURPLE_PLUGIN_MAGIC
)
379 if (plugin
->info
->magic
>= 2 && plugin
->info
->magic
<= 4)
381 struct _PurplePluginInfo2
383 unsigned int api_version
;
384 PurplePluginType type
;
385 char *ui_requirement
;
388 PurplePluginPriority priority
;
398 gboolean (*load
)(PurplePlugin
*plugin
);
399 gboolean (*unload
)(PurplePlugin
*plugin
);
400 void (*destroy
)(PurplePlugin
*plugin
);
404 PurplePluginUiInfo
*prefs_info
;
405 GList
*(*actions
)(PurplePlugin
*plugin
, gpointer context
);
406 } *info2
= (struct _PurplePluginInfo2
*)plugin
->info
;
408 /* This leaks... but only for ancient plugins, so deal with it. */
409 plugin
->info
= g_new0(PurplePluginInfo
, 1);
411 /* We don't really need all these to display the plugin info, but
412 * I'm copying them all for good measure. */
413 plugin
->info
->magic
= info2
->api_version
;
414 plugin
->info
->type
= info2
->type
;
415 plugin
->info
->ui_requirement
= info2
->ui_requirement
;
416 plugin
->info
->flags
= info2
->flags
;
417 plugin
->info
->dependencies
= info2
->dependencies
;
418 plugin
->info
->id
= info2
->id
;
419 plugin
->info
->name
= info2
->name
;
420 plugin
->info
->version
= info2
->version
;
421 plugin
->info
->summary
= info2
->summary
;
422 plugin
->info
->description
= info2
->description
;
423 plugin
->info
->author
= info2
->author
;
424 plugin
->info
->homepage
= info2
->homepage
;
425 plugin
->info
->load
= info2
->load
;
426 plugin
->info
->unload
= info2
->unload
;
427 plugin
->info
->destroy
= info2
->destroy
;
428 plugin
->info
->ui_info
= info2
->ui_info
;
429 plugin
->info
->extra_info
= info2
->extra_info
;
431 if (info2
->api_version
>= 3)
432 plugin
->info
->prefs_info
= info2
->prefs_info
;
434 if (info2
->api_version
>= 4)
435 plugin
->info
->actions
= info2
->actions
;
438 plugin
->error
= g_strdup_printf(_("Plugin magic mismatch %d (need %d)"),
439 plugin
->info
->magic
, PURPLE_PLUGIN_MAGIC
);
440 purple_debug_error("plugins", "%s is not loadable: Plugin magic mismatch %d (need %d)\n",
441 plugin
->path
, plugin
->info
->magic
, PURPLE_PLUGIN_MAGIC
);
442 plugin
->unloadable
= TRUE
;
446 purple_debug_error("plugins", "%s is not loadable: Plugin magic mismatch %d (need %d)\n",
447 plugin
->path
, plugin
->info
->magic
, PURPLE_PLUGIN_MAGIC
);
448 purple_plugin_destroy(plugin
);
452 if (plugin
->info
->major_version
!= PURPLE_MAJOR_VERSION
||
453 plugin
->info
->minor_version
> PURPLE_MINOR_VERSION
)
455 plugin
->error
= g_strdup_printf(_("ABI version mismatch %d.%d.x (need %d.%d.x)"),
456 plugin
->info
->major_version
, plugin
->info
->minor_version
,
457 PURPLE_MAJOR_VERSION
, PURPLE_MINOR_VERSION
);
458 purple_debug_error("plugins", "%s is not loadable: ABI version mismatch %d.%d.x (need %d.%d.x)\n",
459 plugin
->path
, plugin
->info
->major_version
, plugin
->info
->minor_version
,
460 PURPLE_MAJOR_VERSION
, PURPLE_MINOR_VERSION
);
461 plugin
->unloadable
= TRUE
;
465 if (plugin
->info
->type
== PURPLE_PLUGIN_PROTOCOL
)
467 /* If plugin is a PRPL, make sure it implements the required functions */
468 if ((PURPLE_PLUGIN_PROTOCOL_INFO(plugin
)->list_icon
== NULL
) ||
469 (PURPLE_PLUGIN_PROTOCOL_INFO(plugin
)->login
== NULL
) ||
470 (PURPLE_PLUGIN_PROTOCOL_INFO(plugin
)->close
== NULL
))
472 plugin
->error
= g_strdup(_("Plugin does not implement all required functions (list_icon, login and close)"));
473 purple_debug_error("plugins", "%s is not loadable: %s\n",
474 plugin
->path
, plugin
->error
);
475 plugin
->unloadable
= TRUE
;
479 /* For debugging, let's warn about prpl prefs. */
480 if (plugin
->info
->prefs_info
!= NULL
)
482 purple_debug_error("plugins", "%s has a prefs_info, but is a prpl. This is no longer supported.\n",
490 #endif /* !PURPLE_PLUGINS */
493 #ifdef PURPLE_PLUGINS
495 compare_plugins(gconstpointer a
, gconstpointer b
)
497 const PurplePlugin
*plugina
= a
;
498 const PurplePlugin
*pluginb
= b
;
500 return strcmp(plugina
->info
->name
, pluginb
->info
->name
);
502 #endif /* PURPLE_PLUGINS */
505 purple_plugin_load(PurplePlugin
*plugin
)
507 #ifdef PURPLE_PLUGINS
508 GList
*dep_list
= NULL
;
511 g_return_val_if_fail(plugin
!= NULL
, FALSE
);
513 if (purple_plugin_is_loaded(plugin
))
516 if (purple_plugin_is_unloadable(plugin
))
519 g_return_val_if_fail(plugin
->error
== NULL
, FALSE
);
522 * Go through the list of the plugin's dependencies.
524 * First pass: Make sure all the plugins needed are probed.
526 for (l
= plugin
->info
->dependencies
; l
!= NULL
; l
= l
->next
)
528 const char *dep_name
= (const char *)l
->data
;
529 PurplePlugin
*dep_plugin
;
531 dep_plugin
= purple_plugins_find_with_id(dep_name
);
533 if (dep_plugin
== NULL
)
537 tmp
= g_strdup_printf(_("The required plugin %s was not found. "
538 "Please install this plugin and try again."),
541 purple_notify_error(NULL
, NULL
,
542 _("Unable to load the plugin"), tmp
);
545 g_list_free(dep_list
);
550 dep_list
= g_list_append(dep_list
, dep_plugin
);
553 /* Second pass: load all the required plugins. */
554 for (l
= dep_list
; l
!= NULL
; l
= l
->next
)
556 PurplePlugin
*dep_plugin
= (PurplePlugin
*)l
->data
;
558 if (!purple_plugin_is_loaded(dep_plugin
))
560 if (!purple_plugin_load(dep_plugin
))
564 tmp
= g_strdup_printf(_("The required plugin %s was unable to load."),
567 purple_notify_error(NULL
, NULL
,
568 _("Unable to load your plugin."), tmp
);
571 g_list_free(dep_list
);
578 /* Third pass: note that other plugins are dependencies of this plugin.
579 * This is done separately in case we had to bail out earlier. */
580 for (l
= dep_list
; l
!= NULL
; l
= l
->next
)
582 PurplePlugin
*dep_plugin
= (PurplePlugin
*)l
->data
;
583 dep_plugin
->dependent_plugins
= g_list_prepend(dep_plugin
->dependent_plugins
, plugin
->info
->id
);
586 g_list_free(dep_list
);
588 if (plugin
->native_plugin
)
590 if (plugin
->info
->load
!= NULL
&& !plugin
->info
->load(plugin
))
594 PurplePlugin
*loader
;
595 PurplePluginLoaderInfo
*loader_info
;
597 loader
= find_loader_for_plugin(plugin
);
602 loader_info
= PURPLE_PLUGIN_LOADER_INFO(loader
);
604 if (loader_info
->load
!= NULL
)
606 if (!loader_info
->load(plugin
))
611 loaded_plugins
= g_list_insert_sorted(loaded_plugins
, plugin
, compare_plugins
);
613 plugin
->loaded
= TRUE
;
616 load_cb(plugin
, load_cb_data
);
618 purple_signal_emit(purple_plugins_get_handle(), "plugin-load", plugin
);
624 #endif /* !PURPLE_PLUGINS */
628 purple_plugin_unload(PurplePlugin
*plugin
)
630 #ifdef PURPLE_PLUGINS
634 g_return_val_if_fail(plugin
!= NULL
, FALSE
);
635 g_return_val_if_fail(purple_plugin_is_loaded(plugin
), FALSE
);
637 purple_debug_info("plugins", "Unloading plugin %s\n", plugin
->info
->name
);
639 /* Unload all plugins that depend on this plugin. */
640 for (l
= plugin
->dependent_plugins
; l
!= NULL
; l
= ll
) {
641 const char * dep_name
= (const char *)l
->data
;
642 PurplePlugin
*dep_plugin
;
644 /* Store a pointer to the next element in the list.
645 * This is because we'll be modifying this list in the loop. */
648 dep_plugin
= purple_plugins_find_with_id(dep_name
);
650 if (dep_plugin
!= NULL
&& purple_plugin_is_loaded(dep_plugin
))
652 if (!purple_plugin_unload(dep_plugin
))
654 g_free(plugin
->error
);
655 plugin
->error
= g_strdup_printf(_("%s requires %s, but it failed to unload."),
656 _(plugin
->info
->name
),
657 _(dep_plugin
->info
->name
));
663 /* This isn't necessary. This has already been done when unloading dep_plugin. */
664 plugin
->dependent_plugins
= g_list_delete_link(plugin
->dependent_plugins
, l
);
670 /* Remove this plugin from each dependency's dependent_plugins list. */
671 for (l
= plugin
->info
->dependencies
; l
!= NULL
; l
= l
->next
)
673 const char *dep_name
= (const char *)l
->data
;
674 PurplePlugin
*dependency
;
676 dependency
= purple_plugins_find_with_id(dep_name
);
678 if (dependency
!= NULL
)
679 dependency
->dependent_plugins
= g_list_remove(dependency
->dependent_plugins
, plugin
->info
->id
);
681 purple_debug_error("plugins", "Unable to remove from dependency list for %s\n", dep_name
);
684 if (plugin
->native_plugin
) {
685 if (plugin
->info
->unload
&& !plugin
->info
->unload(plugin
))
688 if (plugin
->info
->type
== PURPLE_PLUGIN_PROTOCOL
) {
689 PurplePluginProtocolInfo
*prpl_info
;
692 prpl_info
= PURPLE_PLUGIN_PROTOCOL_INFO(plugin
);
694 for (l
= prpl_info
->user_splits
; l
!= NULL
; l
= l
->next
)
695 purple_account_user_split_destroy(l
->data
);
697 for (l
= prpl_info
->protocol_options
; l
!= NULL
; l
= l
->next
)
698 purple_account_option_destroy(l
->data
);
700 if (prpl_info
->user_splits
!= NULL
) {
701 g_list_free(prpl_info
->user_splits
);
702 prpl_info
->user_splits
= NULL
;
705 if (prpl_info
->protocol_options
!= NULL
) {
706 g_list_free(prpl_info
->protocol_options
);
707 prpl_info
->protocol_options
= NULL
;
711 PurplePlugin
*loader
;
712 PurplePluginLoaderInfo
*loader_info
;
714 loader
= find_loader_for_plugin(plugin
);
719 loader_info
= PURPLE_PLUGIN_LOADER_INFO(loader
);
721 if (loader_info
->unload
&& !loader_info
->unload(plugin
))
725 /* cancel any pending dialogs the plugin has */
726 purple_request_close_with_handle(plugin
);
727 purple_notify_close_with_handle(plugin
);
729 purple_signals_disconnect_by_handle(plugin
);
730 purple_plugin_ipc_unregister_all(plugin
);
732 loaded_plugins
= g_list_remove(loaded_plugins
, plugin
);
733 if ((plugin
->info
!= NULL
) && PURPLE_IS_PROTOCOL_PLUGIN(plugin
))
734 protocol_plugins
= g_list_remove(protocol_plugins
, plugin
);
735 plugins_to_disable
= g_list_remove(plugins_to_disable
, plugin
);
736 plugin
->loaded
= FALSE
;
738 /* We wouldn't be anywhere near here if the plugin wasn't loaded, so
739 * if plugin->error is set at all, it had to be from a previous
740 * unload failure. It's obviously okay now.
742 g_free(plugin
->error
);
743 plugin
->error
= NULL
;
745 if (unload_cb
!= NULL
)
746 unload_cb(plugin
, unload_cb_data
);
748 purple_signal_emit(purple_plugins_get_handle(), "plugin-unload", plugin
);
750 purple_prefs_disconnect_by_handle(plugin
);
755 #endif /* PURPLE_PLUGINS */
759 purple_plugin_disable(PurplePlugin
*plugin
)
761 #ifdef PURPLE_PLUGINS
762 g_return_if_fail(plugin
!= NULL
);
764 if (!g_list_find(plugins_to_disable
, plugin
))
765 plugins_to_disable
= g_list_prepend(plugins_to_disable
, plugin
);
770 purple_plugin_reload(PurplePlugin
*plugin
)
772 #ifdef PURPLE_PLUGINS
773 g_return_val_if_fail(plugin
!= NULL
, FALSE
);
774 g_return_val_if_fail(purple_plugin_is_loaded(plugin
), FALSE
);
776 if (!purple_plugin_unload(plugin
))
779 if (!purple_plugin_load(plugin
))
785 #endif /* !PURPLE_PLUGINS */
789 purple_plugin_destroy(PurplePlugin
*plugin
)
791 #ifdef PURPLE_PLUGINS
792 g_return_if_fail(plugin
!= NULL
);
794 if (purple_plugin_is_loaded(plugin
))
795 purple_plugin_unload(plugin
);
797 plugins
= g_list_remove(plugins
, plugin
);
799 if (load_queue
!= NULL
)
800 load_queue
= g_list_remove(load_queue
, plugin
);
802 /* true, this may leak a little memory if there is a major version
803 * mismatch, but it's a lot better than trying to free something
804 * we shouldn't, and crashing while trying to load an old plugin */
805 if(plugin
->info
== NULL
|| plugin
->info
->magic
!= PURPLE_PLUGIN_MAGIC
||
806 plugin
->info
->major_version
!= PURPLE_MAJOR_VERSION
)
809 g_module_close(plugin
->handle
);
811 g_free(plugin
->path
);
812 g_free(plugin
->error
);
814 PURPLE_DBUS_UNREGISTER_POINTER(plugin
);
820 if (plugin
->info
!= NULL
)
821 g_list_free(plugin
->info
->dependencies
);
823 if (plugin
->native_plugin
)
825 if (plugin
->info
!= NULL
&& plugin
->info
->type
== PURPLE_PLUGIN_LOADER
)
827 PurplePluginLoaderInfo
*loader_info
;
828 GList
*exts
, *l
, *next_l
;
831 loader_info
= PURPLE_PLUGIN_LOADER_INFO(plugin
);
833 if (loader_info
!= NULL
&& loader_info
->exts
!= NULL
)
835 for (exts
= PURPLE_PLUGIN_LOADER_INFO(plugin
)->exts
;
839 for (l
= purple_plugins_get_all(); l
!= NULL
; l
= next_l
)
845 if (p2
->path
!= NULL
&&
846 has_file_extension(p2
->path
, exts
->data
))
848 purple_plugin_destroy(p2
);
853 g_list_free(loader_info
->exts
);
854 loader_info
->exts
= NULL
;
857 plugin_loaders
= g_list_remove(plugin_loaders
, plugin
);
860 if (plugin
->info
!= NULL
&& plugin
->info
->destroy
!= NULL
)
861 plugin
->info
->destroy(plugin
);
864 * I find it extremely useful to do this when using valgrind, as
865 * it keeps all the plugins open, meaning that valgrind is able to
866 * resolve symbol names in leak traces from plugins.
868 if (!g_getenv("PURPLE_LEAKCHECK_HELP") && !RUNNING_ON_VALGRIND
)
870 if (plugin
->handle
!= NULL
)
871 g_module_close(plugin
->handle
);
876 PurplePlugin
*loader
;
877 PurplePluginLoaderInfo
*loader_info
;
879 loader
= find_loader_for_plugin(plugin
);
883 loader_info
= PURPLE_PLUGIN_LOADER_INFO(loader
);
885 if (loader_info
->destroy
!= NULL
)
886 loader_info
->destroy(plugin
);
890 g_free(plugin
->path
);
891 g_free(plugin
->error
);
893 PURPLE_DBUS_UNREGISTER_POINTER(plugin
);
896 #endif /* !PURPLE_PLUGINS */
900 purple_plugin_is_loaded(const PurplePlugin
*plugin
)
902 g_return_val_if_fail(plugin
!= NULL
, FALSE
);
904 return plugin
->loaded
;
908 purple_plugin_is_unloadable(const PurplePlugin
*plugin
)
910 g_return_val_if_fail(plugin
!= NULL
, FALSE
);
912 return plugin
->unloadable
;
916 purple_plugin_get_id(const PurplePlugin
*plugin
) {
917 g_return_val_if_fail(plugin
, NULL
);
918 g_return_val_if_fail(plugin
->info
, NULL
);
920 return plugin
->info
->id
;
924 purple_plugin_get_name(const PurplePlugin
*plugin
) {
925 g_return_val_if_fail(plugin
, NULL
);
926 g_return_val_if_fail(plugin
->info
, NULL
);
928 return _(plugin
->info
->name
);
932 purple_plugin_get_version(const PurplePlugin
*plugin
) {
933 g_return_val_if_fail(plugin
, NULL
);
934 g_return_val_if_fail(plugin
->info
, NULL
);
936 return plugin
->info
->version
;
940 purple_plugin_get_summary(const PurplePlugin
*plugin
) {
941 g_return_val_if_fail(plugin
, NULL
);
942 g_return_val_if_fail(plugin
->info
, NULL
);
944 return _(plugin
->info
->summary
);
948 purple_plugin_get_description(const PurplePlugin
*plugin
) {
949 g_return_val_if_fail(plugin
, NULL
);
950 g_return_val_if_fail(plugin
->info
, NULL
);
952 return _(plugin
->info
->description
);
956 purple_plugin_get_author(const PurplePlugin
*plugin
) {
957 g_return_val_if_fail(plugin
, NULL
);
958 g_return_val_if_fail(plugin
->info
, NULL
);
960 return _(plugin
->info
->author
);
964 purple_plugin_get_homepage(const PurplePlugin
*plugin
) {
965 g_return_val_if_fail(plugin
, NULL
);
966 g_return_val_if_fail(plugin
->info
, NULL
);
968 return plugin
->info
->homepage
;
971 /**************************************************************************
973 **************************************************************************/
975 destroy_ipc_info(void *data
)
977 PurplePluginIpcCommand
*ipc_command
= (PurplePluginIpcCommand
*)data
;
980 if (ipc_command
->params
!= NULL
)
982 for (i
= 0; i
< ipc_command
->num_params
; i
++)
983 purple_value_destroy(ipc_command
->params
[i
]);
985 g_free(ipc_command
->params
);
988 if (ipc_command
->ret_value
!= NULL
)
989 purple_value_destroy(ipc_command
->ret_value
);
995 purple_plugin_ipc_register(PurplePlugin
*plugin
, const char *command
,
996 PurpleCallback func
, PurpleSignalMarshalFunc marshal
,
997 PurpleValue
*ret_value
, int num_params
, ...)
999 PurplePluginIpcInfo
*ipc_info
;
1000 PurplePluginIpcCommand
*ipc_command
;
1002 g_return_val_if_fail(plugin
!= NULL
, FALSE
);
1003 g_return_val_if_fail(command
!= NULL
, FALSE
);
1004 g_return_val_if_fail(func
!= NULL
, FALSE
);
1005 g_return_val_if_fail(marshal
!= NULL
, FALSE
);
1007 if (plugin
->ipc_data
== NULL
)
1009 ipc_info
= plugin
->ipc_data
= g_new0(PurplePluginIpcInfo
, 1);
1010 ipc_info
->commands
= g_hash_table_new_full(g_str_hash
, g_str_equal
,
1011 g_free
, destroy_ipc_info
);
1014 ipc_info
= (PurplePluginIpcInfo
*)plugin
->ipc_data
;
1016 ipc_command
= g_new0(PurplePluginIpcCommand
, 1);
1017 ipc_command
->func
= func
;
1018 ipc_command
->marshal
= marshal
;
1019 ipc_command
->num_params
= num_params
;
1020 ipc_command
->ret_value
= ret_value
;
1027 ipc_command
->params
= g_new0(PurpleValue
*, num_params
);
1029 va_start(args
, num_params
);
1031 for (i
= 0; i
< num_params
; i
++)
1032 ipc_command
->params
[i
] = va_arg(args
, PurpleValue
*);
1037 g_hash_table_replace(ipc_info
->commands
, g_strdup(command
), ipc_command
);
1039 ipc_info
->command_count
++;
1045 purple_plugin_ipc_unregister(PurplePlugin
*plugin
, const char *command
)
1047 PurplePluginIpcInfo
*ipc_info
;
1049 g_return_if_fail(plugin
!= NULL
);
1050 g_return_if_fail(command
!= NULL
);
1052 ipc_info
= (PurplePluginIpcInfo
*)plugin
->ipc_data
;
1054 if (ipc_info
== NULL
||
1055 g_hash_table_lookup(ipc_info
->commands
, command
) == NULL
)
1057 purple_debug_error("plugins",
1058 "IPC command '%s' was not registered for plugin %s\n",
1059 command
, plugin
->info
->name
);
1063 g_hash_table_remove(ipc_info
->commands
, command
);
1065 ipc_info
->command_count
--;
1067 if (ipc_info
->command_count
== 0)
1069 g_hash_table_destroy(ipc_info
->commands
);
1072 plugin
->ipc_data
= NULL
;
1077 purple_plugin_ipc_unregister_all(PurplePlugin
*plugin
)
1079 PurplePluginIpcInfo
*ipc_info
;
1081 g_return_if_fail(plugin
!= NULL
);
1083 if (plugin
->ipc_data
== NULL
)
1084 return; /* Silently ignore it. */
1086 ipc_info
= (PurplePluginIpcInfo
*)plugin
->ipc_data
;
1088 g_hash_table_destroy(ipc_info
->commands
);
1091 plugin
->ipc_data
= NULL
;
1095 purple_plugin_ipc_get_params(PurplePlugin
*plugin
, const char *command
,
1096 PurpleValue
**ret_value
, int *num_params
,
1097 PurpleValue
***params
)
1099 PurplePluginIpcInfo
*ipc_info
;
1100 PurplePluginIpcCommand
*ipc_command
;
1102 g_return_val_if_fail(plugin
!= NULL
, FALSE
);
1103 g_return_val_if_fail(command
!= NULL
, FALSE
);
1105 ipc_info
= (PurplePluginIpcInfo
*)plugin
->ipc_data
;
1107 if (ipc_info
== NULL
||
1108 (ipc_command
= g_hash_table_lookup(ipc_info
->commands
,
1111 purple_debug_error("plugins",
1112 "IPC command '%s' was not registered for plugin %s\n",
1113 command
, plugin
->info
->name
);
1118 if (num_params
!= NULL
)
1119 *num_params
= ipc_command
->num_params
;
1122 *params
= ipc_command
->params
;
1124 if (ret_value
!= NULL
)
1125 *ret_value
= ipc_command
->ret_value
;
1131 purple_plugin_ipc_call(PurplePlugin
*plugin
, const char *command
,
1134 PurplePluginIpcInfo
*ipc_info
;
1135 PurplePluginIpcCommand
*ipc_command
;
1142 g_return_val_if_fail(plugin
!= NULL
, NULL
);
1143 g_return_val_if_fail(command
!= NULL
, NULL
);
1145 ipc_info
= (PurplePluginIpcInfo
*)plugin
->ipc_data
;
1147 if (ipc_info
== NULL
||
1148 (ipc_command
= g_hash_table_lookup(ipc_info
->commands
,
1151 purple_debug_error("plugins",
1152 "IPC command '%s' was not registered for plugin %s\n",
1153 command
, plugin
->info
->name
);
1159 ipc_command
->marshal(ipc_command
->func
, args
, NULL
, &ret_value
);
1168 /**************************************************************************
1170 **************************************************************************/
1172 purple_plugins_get_handle(void) {
1179 purple_plugins_init(void) {
1180 void *handle
= purple_plugins_get_handle();
1182 purple_plugins_add_search_path(LIBDIR
);
1184 purple_signal_register(handle
, "plugin-load",
1185 purple_marshal_VOID__POINTER
,
1187 purple_value_new(PURPLE_TYPE_SUBTYPE
,
1188 PURPLE_SUBTYPE_PLUGIN
));
1189 purple_signal_register(handle
, "plugin-unload",
1190 purple_marshal_VOID__POINTER
,
1192 purple_value_new(PURPLE_TYPE_SUBTYPE
,
1193 PURPLE_SUBTYPE_PLUGIN
));
1197 purple_plugins_uninit(void)
1199 void *handle
= purple_plugins_get_handle();
1201 purple_signals_disconnect_by_handle(handle
);
1202 purple_signals_unregister_by_instance(handle
);
1204 while (search_paths
) {
1205 g_free(search_paths
->data
);
1206 search_paths
= g_list_delete_link(search_paths
, search_paths
);
1210 /**************************************************************************
1212 **************************************************************************/
1214 purple_plugins_add_search_path(const char *path
)
1216 g_return_if_fail(path
!= NULL
);
1218 if (g_list_find_custom(search_paths
, path
, (GCompareFunc
)strcmp
))
1221 search_paths
= g_list_append(search_paths
, g_strdup(path
));
1225 purple_plugins_get_search_paths()
1227 return search_paths
;
1231 purple_plugins_unload_all(void)
1233 #ifdef PURPLE_PLUGINS
1235 while (loaded_plugins
!= NULL
)
1236 purple_plugin_unload(loaded_plugins
->data
);
1238 #endif /* PURPLE_PLUGINS */
1242 purple_plugins_unload(PurplePluginType type
)
1244 #ifdef PURPLE_PLUGINS
1247 for (l
= plugins
; l
; l
= l
->next
) {
1248 PurplePlugin
*plugin
= l
->data
;
1249 if (plugin
->info
->type
== type
&& purple_plugin_is_loaded(plugin
))
1250 purple_plugin_unload(plugin
);
1253 #endif /* PURPLE_PLUGINS */
1257 purple_plugins_destroy_all(void)
1259 #ifdef PURPLE_PLUGINS
1261 while (plugins
!= NULL
)
1262 purple_plugin_destroy(plugins
->data
);
1264 #endif /* PURPLE_PLUGINS */
1268 purple_plugins_save_loaded(const char *key
)
1270 #ifdef PURPLE_PLUGINS
1272 GList
*files
= NULL
;
1274 for (pl
= purple_plugins_get_loaded(); pl
!= NULL
; pl
= pl
->next
) {
1275 PurplePlugin
*plugin
= pl
->data
;
1277 if (plugin
->info
->type
!= PURPLE_PLUGIN_PROTOCOL
&&
1278 plugin
->info
->type
!= PURPLE_PLUGIN_LOADER
&&
1279 !g_list_find(plugins_to_disable
, plugin
)) {
1280 files
= g_list_append(files
, plugin
->path
);
1284 purple_prefs_set_path_list(key
, files
);
1290 purple_plugins_load_saved(const char *key
)
1292 #ifdef PURPLE_PLUGINS
1295 g_return_if_fail(key
!= NULL
);
1297 files
= purple_prefs_get_path_list(key
);
1299 for (f
= files
; f
; f
= f
->next
)
1303 PurplePlugin
*plugin
;
1305 if (f
->data
== NULL
)
1311 * We don't know if the filename uses Windows or Unix path
1312 * separators (because people might be sharing a prefs.xml
1313 * file across systems), so we find the last occurrence
1316 basename
= strrchr(filename
, '/');
1317 if ((basename
== NULL
) || (basename
< strrchr(filename
, '\\')))
1318 basename
= strrchr(filename
, '\\');
1319 if (basename
!= NULL
)
1322 /* Strip the extension */
1324 basename
= purple_plugin_get_basename(basename
);
1326 if (((plugin
= purple_plugins_find_with_filename(filename
)) != NULL
) ||
1327 (basename
&& (plugin
= purple_plugins_find_with_basename(basename
)) != NULL
) ||
1328 ((plugin
= purple_plugin_probe(filename
)) != NULL
))
1330 purple_debug_info("plugins", "Loading saved plugin %s\n",
1332 purple_plugin_load(plugin
);
1336 purple_debug_error("plugins", "Unable to find saved plugin %s\n",
1346 #endif /* PURPLE_PLUGINS */
1351 purple_plugins_probe(const char *ext
)
1353 #ifdef PURPLE_PLUGINS
1357 PurplePlugin
*plugin
;
1359 const char *search_path
;
1361 if (!g_module_supported())
1365 for (cur
= search_paths
; cur
!= NULL
; cur
= cur
->next
)
1367 search_path
= cur
->data
;
1369 dir
= g_dir_open(search_path
, 0, NULL
);
1373 while ((file
= g_dir_read_name(dir
)) != NULL
)
1375 path
= g_build_filename(search_path
, file
, NULL
);
1377 if (ext
== NULL
|| has_file_extension(file
, ext
))
1378 purple_plugin_probe(path
);
1387 /* See if we have any plugins waiting to load */
1388 while (load_queue
!= NULL
)
1390 plugin
= (PurplePlugin
*)load_queue
->data
;
1392 load_queue
= g_list_remove(load_queue
, plugin
);
1394 if (plugin
== NULL
|| plugin
->info
== NULL
)
1397 if (plugin
->info
->type
== PURPLE_PLUGIN_LOADER
)
1399 /* We'll just load this right now. */
1400 if (!purple_plugin_load(plugin
))
1402 purple_plugin_destroy(plugin
);
1407 plugin_loaders
= g_list_append(plugin_loaders
, plugin
);
1409 for (cur
= PURPLE_PLUGIN_LOADER_INFO(plugin
)->exts
;
1413 purple_plugins_probe(cur
->data
);
1416 else if (plugin
->info
->type
== PURPLE_PLUGIN_PROTOCOL
)
1418 /* We'll just load this right now. */
1419 if (!purple_plugin_load(plugin
))
1421 purple_plugin_destroy(plugin
);
1426 /* Make sure we don't load two PRPLs with the same name? */
1427 if (purple_find_prpl(plugin
->info
->id
))
1429 /* Nothing to see here--move along, move along */
1430 purple_plugin_destroy(plugin
);
1435 protocol_plugins
= g_list_insert_sorted(protocol_plugins
, plugin
,
1436 (GCompareFunc
)compare_prpl
);
1440 if (probe_cb
!= NULL
)
1441 probe_cb(probe_cb_data
);
1443 #endif /* PURPLE_PLUGINS */
1447 purple_plugin_register(PurplePlugin
*plugin
)
1449 g_return_val_if_fail(plugin
!= NULL
, FALSE
);
1451 /* If this plugin has been registered already then exit */
1452 if (g_list_find(plugins
, plugin
))
1455 /* Ensure the plugin has the requisite information */
1456 if (plugin
->info
->type
== PURPLE_PLUGIN_LOADER
)
1458 PurplePluginLoaderInfo
*loader_info
;
1460 loader_info
= PURPLE_PLUGIN_LOADER_INFO(plugin
);
1462 if (loader_info
== NULL
)
1464 purple_debug_error("plugins", "%s is not loadable, loader plugin missing loader_info\n",
1469 else if (plugin
->info
->type
== PURPLE_PLUGIN_PROTOCOL
)
1471 PurplePluginProtocolInfo
*prpl_info
;
1473 prpl_info
= PURPLE_PLUGIN_PROTOCOL_INFO(plugin
);
1475 if (prpl_info
== NULL
)
1477 purple_debug_error("plugins", "%s is not loadable, protocol plugin missing prpl_info\n",
1483 #ifdef PURPLE_PLUGINS
1484 /* This plugin should be probed and maybe loaded--add it to the queue */
1485 load_queue
= g_list_append(load_queue
, plugin
);
1487 if (plugin
->info
!= NULL
)
1489 if (plugin
->info
->type
== PURPLE_PLUGIN_PROTOCOL
)
1490 protocol_plugins
= g_list_insert_sorted(protocol_plugins
, plugin
,
1491 (GCompareFunc
)compare_prpl
);
1492 if (plugin
->info
->load
!= NULL
)
1493 if (!plugin
->info
->load(plugin
))
1498 plugins
= g_list_append(plugins
, plugin
);
1504 purple_plugins_enabled(void)
1506 #ifdef PURPLE_PLUGINS
1514 purple_plugins_register_probe_notify_cb(void (*func
)(void *), void *data
)
1517 probe_cb_data
= data
;
1521 purple_plugins_unregister_probe_notify_cb(void (*func
)(void *))
1524 probe_cb_data
= NULL
;
1528 purple_plugins_register_load_notify_cb(void (*func
)(PurplePlugin
*, void *),
1532 load_cb_data
= data
;
1536 purple_plugins_unregister_load_notify_cb(void (*func
)(PurplePlugin
*, void *))
1539 load_cb_data
= NULL
;
1543 purple_plugins_register_unload_notify_cb(void (*func
)(PurplePlugin
*, void *),
1547 unload_cb_data
= data
;
1551 purple_plugins_unregister_unload_notify_cb(void (*func
)(PurplePlugin
*, void *))
1554 unload_cb_data
= NULL
;
1558 purple_plugins_find_with_name(const char *name
)
1560 PurplePlugin
*plugin
;
1563 for (l
= plugins
; l
!= NULL
; l
= l
->next
) {
1566 if (purple_strequal(plugin
->info
->name
, name
))
1574 purple_plugins_find_with_filename(const char *filename
)
1576 PurplePlugin
*plugin
;
1579 for (l
= plugins
; l
!= NULL
; l
= l
->next
) {
1582 if (purple_strequal(plugin
->path
, filename
))
1590 purple_plugins_find_with_basename(const char *basename
)
1592 #ifdef PURPLE_PLUGINS
1593 PurplePlugin
*plugin
;
1597 g_return_val_if_fail(basename
!= NULL
, NULL
);
1599 for (l
= plugins
; l
!= NULL
; l
= l
->next
)
1601 plugin
= (PurplePlugin
*)l
->data
;
1603 if (plugin
->path
!= NULL
) {
1604 tmp
= purple_plugin_get_basename(plugin
->path
);
1605 if (purple_strequal(tmp
, basename
))
1614 #endif /* PURPLE_PLUGINS */
1620 purple_plugins_find_with_id(const char *id
)
1622 PurplePlugin
*plugin
;
1625 g_return_val_if_fail(id
!= NULL
, NULL
);
1627 for (l
= plugins
; l
!= NULL
; l
= l
->next
)
1631 if (purple_strequal(plugin
->info
->id
, id
))
1639 purple_plugins_get_loaded(void)
1641 return loaded_plugins
;
1645 purple_plugins_get_protocols(void)
1647 return protocol_plugins
;
1651 purple_plugins_get_all(void)
1657 PurplePluginAction
*
1658 purple_plugin_action_new(const char* label
, void (*callback
)(PurplePluginAction
*))
1660 PurplePluginAction
*act
= g_new0(PurplePluginAction
, 1);
1662 act
->label
= g_strdup(label
);
1663 act
->callback
= callback
;
1669 purple_plugin_action_free(PurplePluginAction
*action
)
1671 g_return_if_fail(action
!= NULL
);
1673 g_free(action
->label
);