Mercurial > pidgin
annotate libgaim/plugins/perl/perl.c @ 15015:9dcd88cd7dee
[gaim-migrate @ 17797]
Don't allow the buddy list to get lost when switching "Show system tray icon" to "On Unread messages"
committer: Tailor Script <tailor@pidgin.im>
author | Daniel Atallah <daniel.atallah@gmail.com> |
---|---|
date | Tue, 21 Nov 2006 02:24:19 +0000 |
parents | b1fcd2fc903a |
children |
rev | line source |
---|---|
14192 | 1 /* |
2 * gaim | |
3 * | |
4 * Copyright (C) 2003 Christian Hammond <chipx86@gnupdate.org> | |
5 * | |
6 * This program is free software; you can redistribute it and/or modify | |
7 * it under the terms of the GNU General Public License as published by | |
8 * the Free Software Foundation; either version 2 of the License, or | |
9 * (at your option) any later version. | |
10 * | |
11 * This program is distributed in the hope that it will be useful, | |
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
14 * GNU General Public License for more details. | |
15 * | |
16 * You should have received a copy of the GNU General Public License | |
17 * along with this program; if not, write to the Free Software | |
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | |
19 */ | |
20 #ifdef HAVE_CONFIG_H | |
21 #include <config.h> | |
22 # ifdef HAVE_LIMITS_H | |
23 # include <limits.h> | |
24 # ifndef NAME_MAX | |
25 # define NAME_MAX _POSIX_NAME_MAX | |
26 # endif | |
27 # endif | |
28 #endif | |
29 | |
30 #ifdef DEBUG | |
31 # undef DEBUG | |
32 #endif | |
33 | |
34 #undef PACKAGE | |
35 | |
36 #define group perl_group | |
37 | |
38 #ifdef _WIN32 | |
39 /* This took me an age to figure out.. without this __declspec(dllimport) | |
40 * will be ignored. | |
41 */ | |
42 # define HASATTRIBUTE | |
43 #endif | |
44 | |
45 #include <EXTERN.h> | |
46 | |
47 #ifndef _SEM_SEMUN_UNDEFINED | |
48 # define HAS_UNION_SEMUN | |
49 #endif | |
50 | |
51 #include <perl.h> | |
52 #include <XSUB.h> | |
53 | |
54 #ifndef _WIN32 | |
55 # include <sys/mman.h> | |
56 #endif | |
57 | |
58 #undef PACKAGE | |
59 | |
60 #ifndef _WIN32 | |
61 # include <dirent.h> | |
62 #else | |
63 /* We're using perl's win32 port of this */ | |
64 # define dirent direct | |
65 #endif | |
66 | |
67 #undef group | |
68 | |
69 /* perl module support */ | |
70 #ifdef OLD_PERL | |
71 extern void boot_DynaLoader _((CV * cv)); | |
72 #else | |
73 extern void boot_DynaLoader _((pTHX_ CV * cv)); /* perl is so wacky */ | |
74 #endif | |
75 | |
76 #undef _ | |
77 #ifdef DEBUG | |
78 # undef DEBUG | |
79 #endif | |
80 #ifdef _WIN32 | |
81 # undef pipe | |
82 #endif | |
83 | |
84 #ifdef _WIN32 | |
85 #define _WIN32DEP_H_ | |
86 #endif | |
87 #include "internal.h" | |
88 #include "debug.h" | |
89 #include "plugin.h" | |
90 #include "signals.h" | |
91 #include "version.h" | |
92 | |
93 #include "perl-common.h" | |
94 #include "perl-handlers.h" | |
95 | |
96 #define PERL_PLUGIN_ID "core-perl" | |
97 | |
98 PerlInterpreter *my_perl = NULL; | |
99 | |
100 static GaimPluginUiInfo ui_info = | |
101 { | |
102 gaim_perl_get_plugin_frame, | |
103 0, /* page_num (Reserved) */ | |
104 NULL /* frame (Reserved) */ | |
105 }; | |
106 | |
14364
338ac096e322
[gaim-migrate @ 17070]
Daniel Atallah <daniel.atallah@gmail.com>
parents:
14192
diff
changeset
|
107 #ifdef GAIM_GTKPERL |
14192 | 108 static GaimGtkPluginUiInfo gtk_ui_info = |
109 { | |
110 gaim_perl_gtk_get_plugin_frame, | |
111 0 /* page_num (Reserved) */ | |
112 }; | |
14364
338ac096e322
[gaim-migrate @ 17070]
Daniel Atallah <daniel.atallah@gmail.com>
parents:
14192
diff
changeset
|
113 #endif |
14192 | 114 |
115 static void | |
116 #ifdef OLD_PERL | |
117 xs_init() | |
118 #else | |
119 xs_init(pTHX) | |
120 #endif | |
121 { | |
122 char *file = __FILE__; | |
123 | |
124 /* This one allows dynamic loading of perl modules in perl scripts by | |
125 * the 'use perlmod;' construction */ | |
126 newXS("DynaLoader::boot_DynaLoader", boot_DynaLoader, file); | |
127 } | |
128 | |
129 static void | |
130 perl_init(void) | |
131 { | |
132 /* changed the name of the variable from load_file to perl_definitions | |
133 * since now it does much more than defining the load_file sub. | |
134 * Moreover, deplaced the initialisation to the xs_init function. | |
135 * (TheHobbit) */ | |
136 char *perl_args[] = { "", "-e", "0", "-w" }; | |
137 char perl_definitions[] = | |
138 { | |
139 /* We use to function one to load a file the other to execute | |
140 * the string obtained from the first and holding the file | |
141 * contents. This allows to have a really local $/ without | |
142 * introducing temp variables to hold the old value. Just a | |
143 * question of style:) */ | |
144 "package Gaim::PerlLoader;" | |
145 "use Symbol;" | |
146 | |
147 "sub load_file {" | |
148 "my $f_name=shift;" | |
149 "local $/=undef;" | |
150 "open FH,$f_name or return \"__FAILED__\";" | |
151 "$_=<FH>;" | |
152 "close FH;" | |
153 "return $_;" | |
154 "}" | |
155 | |
156 "sub destroy_package {" | |
157 "eval { $_[0]->UNLOAD() if $_[0]->can('UNLOAD'); };" | |
158 "Symbol::delete_package($_[0]);" | |
159 "}" | |
160 | |
161 "sub load_n_eval {" | |
162 "my ($f_name, $package) = @_;" | |
163 "destroy_package($package);" | |
164 "my $strin=load_file($f_name);" | |
165 "return 2 if($strin eq \"__FAILED__\");" | |
166 "my $eval = qq{package $package; $strin;};" | |
167 | |
168 "{" | |
169 " eval $eval;" | |
170 "}" | |
171 | |
172 "if($@) {" | |
173 /*" #something went wrong\n"*/ | |
174 "die(\"Errors loading file $f_name: $@\");" | |
175 "}" | |
176 | |
177 "return 0;" | |
178 "}" | |
179 }; | |
180 | |
181 my_perl = perl_alloc(); | |
182 PERL_SET_CONTEXT(my_perl); | |
183 PL_perl_destruct_level = 1; | |
184 perl_construct(my_perl); | |
185 #ifdef DEBUG | |
186 perl_parse(my_perl, xs_init, 4, perl_args, NULL); | |
187 #else | |
188 perl_parse(my_perl, xs_init, 3, perl_args, NULL); | |
189 #endif | |
190 #ifdef HAVE_PERL_EVAL_PV | |
191 eval_pv(perl_definitions, TRUE); | |
192 #else | |
193 perl_eval_pv(perl_definitions, TRUE); /* deprecated */ | |
194 #endif | |
195 perl_run(my_perl); | |
196 } | |
197 | |
198 static void | |
199 perl_end(void) | |
200 { | |
201 if (my_perl == NULL) | |
202 return; | |
203 | |
204 PL_perl_destruct_level = 1; | |
205 PERL_SET_CONTEXT(my_perl); | |
206 perl_eval_pv( | |
207 "foreach my $lib (@DynaLoader::dl_modules) {" | |
208 "if ($lib =~ /^Gaim\\b/) {" | |
209 "$lib .= '::deinit();';" | |
210 "eval $lib;" | |
211 "}" | |
212 "}", | |
213 TRUE); | |
214 | |
215 PL_perl_destruct_level = 1; | |
216 PERL_SET_CONTEXT(my_perl); | |
217 perl_destruct(my_perl); | |
218 perl_free(my_perl); | |
219 my_perl = NULL; | |
220 } | |
221 | |
222 void | |
223 gaim_perl_callXS(void (*subaddr)(pTHX_ CV *cv), CV *cv, SV **mark) | |
224 { | |
225 dSP; | |
226 | |
227 PUSHMARK(mark); | |
228 (*subaddr)(aTHX_ cv); | |
229 | |
230 PUTBACK; | |
231 } | |
232 | |
233 static gboolean | |
234 probe_perl_plugin(GaimPlugin *plugin) | |
235 { | |
236 /* XXX This would be much faster if I didn't create a new | |
237 * PerlInterpreter every time I probed a plugin */ | |
238 | |
239 PerlInterpreter *prober = perl_alloc(); | |
240 char *argv[] = {"", plugin->path }; | |
241 gboolean status = TRUE; | |
242 HV *plugin_info; | |
243 PERL_SET_CONTEXT(prober); | |
244 PL_perl_destruct_level = 1; | |
245 perl_construct(prober); | |
246 | |
247 perl_parse(prober, xs_init, 2, argv, NULL); | |
248 | |
249 perl_run(prober); | |
250 | |
251 plugin_info = perl_get_hv("PLUGIN_INFO", FALSE); | |
252 | |
253 if (plugin_info == NULL) | |
254 status = FALSE; | |
255 else if (!hv_exists(plugin_info, "perl_api_version", | |
256 strlen("perl_api_version")) || | |
257 !hv_exists(plugin_info, "name", strlen("name")) || | |
258 !hv_exists(plugin_info, "load", strlen("load"))) { | |
259 /* Not a valid plugin. */ | |
260 | |
261 status = FALSE; | |
262 } else { | |
263 SV **key; | |
264 int perl_api_ver; | |
265 | |
266 key = hv_fetch(plugin_info, "perl_api_version", | |
267 strlen("perl_api_version"), 0); | |
268 | |
269 perl_api_ver = SvIV(*key); | |
270 | |
271 if (perl_api_ver != 2) | |
272 status = FALSE; | |
273 else { | |
274 GaimPluginInfo *info; | |
275 GaimPerlScript *gps; | |
276 char *basename; | |
277 STRLEN len; | |
278 | |
279 info = g_new0(GaimPluginInfo, 1); | |
280 gps = g_new0(GaimPerlScript, 1); | |
281 | |
282 info->magic = GAIM_PLUGIN_MAGIC; | |
283 info->major_version = GAIM_MAJOR_VERSION; | |
284 info->minor_version = GAIM_MINOR_VERSION; | |
285 info->type = GAIM_PLUGIN_STANDARD; | |
286 | |
287 info->dependencies = g_list_append(info->dependencies, | |
288 PERL_PLUGIN_ID); | |
289 | |
290 gps->plugin = plugin; | |
291 | |
292 basename = g_path_get_basename(plugin->path); | |
293 gaim_perl_normalize_script_name(basename); | |
294 gps->package = g_strdup_printf("Gaim::Script::%s", | |
295 basename); | |
296 g_free(basename); | |
297 | |
298 /* We know this one exists. */ | |
299 key = hv_fetch(plugin_info, "name", strlen("name"), 0); | |
300 info->name = g_strdup(SvPV(*key, len)); | |
301 /* Set id here in case we don't find one later. */ | |
302 info->id = g_strdup(SvPV(*key, len)); | |
303 | |
14372
d5c22258df09
[gaim-migrate @ 17078]
Etan Reisner <pidgin@unreliablesource.net>
parents:
14364
diff
changeset
|
304 #ifdef GAIM_GTKPERL |
14192 | 305 if ((key = hv_fetch(plugin_info, "GTK_UI", |
306 strlen("GTK_UI"), 0))) | |
307 info->ui_requirement = GAIM_GTK_PLUGIN_TYPE; | |
14372
d5c22258df09
[gaim-migrate @ 17078]
Etan Reisner <pidgin@unreliablesource.net>
parents:
14364
diff
changeset
|
308 #endif |
14192 | 309 |
310 if ((key = hv_fetch(plugin_info, "url", | |
311 strlen("url"), 0))) | |
312 info->homepage = g_strdup(SvPV(*key, len)); | |
313 | |
314 if ((key = hv_fetch(plugin_info, "author", | |
315 strlen("author"), 0))) | |
316 info->author = g_strdup(SvPV(*key, len)); | |
317 | |
318 if ((key = hv_fetch(plugin_info, "summary", | |
319 strlen("summary"), 0))) | |
320 info->summary = g_strdup(SvPV(*key, len)); | |
321 | |
322 if ((key = hv_fetch(plugin_info, "description", | |
323 strlen("description"), 0))) | |
324 info->description = g_strdup(SvPV(*key, len)); | |
325 | |
326 if ((key = hv_fetch(plugin_info, "version", | |
327 strlen("version"), 0))) | |
328 info->version = g_strdup(SvPV(*key, len)); | |
329 | |
330 /* We know this one exists. */ | |
331 key = hv_fetch(plugin_info, "load", strlen("load"), 0); | |
332 gps->load_sub = g_strdup_printf("%s::%s", gps->package, | |
333 SvPV(*key, len)); | |
334 | |
335 if ((key = hv_fetch(plugin_info, "unload", | |
336 strlen("unload"), 0))) | |
337 gps->unload_sub = g_strdup_printf("%s::%s", | |
338 gps->package, | |
339 SvPV(*key, len)); | |
340 | |
341 if ((key = hv_fetch(plugin_info, "id", | |
342 strlen("id"), 0))) { | |
343 g_free(info->id); | |
344 info->id = g_strdup_printf("perl-%s", | |
345 SvPV(*key, len)); | |
346 } | |
347 | |
348 /********************************************************/ | |
349 /* Only one of the next two options should be present */ | |
350 /* */ | |
351 /* prefs_info - Uses non-GUI (read GTK) gaim API calls */ | |
352 /* and creates a GaimPluginPrefInfo type. */ | |
353 /* */ | |
354 /* gtk_prefs_info - Requires gtk2-perl be installed by */ | |
355 /* the user and he must create a */ | |
356 /* GtkWidget the user and he must */ | |
357 /* create a GtkWidget representing the */ | |
358 /* plugin preferences page. */ | |
359 /********************************************************/ | |
360 if ((key = hv_fetch(plugin_info, "prefs_info", | |
361 strlen("prefs_info"), 0))) { | |
362 /* key now is the name of the Perl sub that | |
363 * will create a frame for us */ | |
364 gps->prefs_sub = g_strdup_printf("%s::%s", | |
365 gps->package, | |
366 SvPV(*key, len)); | |
367 info->prefs_info = &ui_info; | |
368 } | |
369 | |
14364
338ac096e322
[gaim-migrate @ 17070]
Daniel Atallah <daniel.atallah@gmail.com>
parents:
14192
diff
changeset
|
370 #ifdef GAIM_GTKPERL |
14192 | 371 if ((key = hv_fetch(plugin_info, "gtk_prefs_info", |
372 strlen("gtk_prefs_info"), 0))) { | |
373 /* key now is the name of the Perl sub that | |
374 * will create a frame for us */ | |
375 gps->gtk_prefs_sub = g_strdup_printf("%s::%s", | |
376 gps->package, | |
377 SvPV(*key, len)); | |
378 info->ui_info = >k_ui_info; | |
379 } | |
14364
338ac096e322
[gaim-migrate @ 17070]
Daniel Atallah <daniel.atallah@gmail.com>
parents:
14192
diff
changeset
|
380 #endif |
14192 | 381 |
382 if ((key = hv_fetch(plugin_info, "plugin_action_sub", | |
383 strlen("plugin_action_sub"), 0))) { | |
384 gps->plugin_action_sub = g_strdup_printf("%s::%s", | |
385 gps->package, | |
386 SvPV(*key, len)); | |
387 info->actions = gaim_perl_plugin_actions; | |
388 } | |
389 | |
390 plugin->info = info; | |
391 info->extra_info = gps; | |
392 | |
393 status = gaim_plugin_register(plugin); | |
394 } | |
395 } | |
396 | |
397 PL_perl_destruct_level = 1; | |
398 PERL_SET_CONTEXT(prober); | |
399 perl_destruct(prober); | |
400 perl_free(prober); | |
401 return status; | |
402 } | |
403 | |
404 static gboolean | |
405 load_perl_plugin(GaimPlugin *plugin) | |
406 { | |
407 GaimPerlScript *gps = (GaimPerlScript *)plugin->info->extra_info; | |
408 char *atmp[3] = { plugin->path, NULL, NULL }; | |
409 | |
410 if (gps == NULL || gps->load_sub == NULL) | |
411 return FALSE; | |
412 | |
413 gaim_debug(GAIM_DEBUG_INFO, "perl", "Loading perl script\n"); | |
414 | |
415 if (my_perl == NULL) | |
416 perl_init(); | |
417 | |
418 plugin->handle = gps; | |
419 | |
420 atmp[1] = gps->package; | |
421 | |
422 PERL_SET_CONTEXT(my_perl); | |
423 execute_perl("Gaim::PerlLoader::load_n_eval", 2, atmp); | |
424 | |
425 { | |
426 dSP; | |
427 PERL_SET_CONTEXT(my_perl); | |
428 SPAGAIN; | |
429 ENTER; | |
430 SAVETMPS; | |
431 PUSHMARK(sp); | |
432 XPUSHs(sv_2mortal(gaim_perl_bless_object(plugin, | |
433 "Gaim::Plugin"))); | |
434 PUTBACK; | |
435 | |
436 perl_call_pv(gps->load_sub, G_EVAL | G_SCALAR); | |
437 SPAGAIN; | |
438 | |
439 if (SvTRUE(ERRSV)) { | |
440 STRLEN len; | |
441 | |
442 gaim_debug(GAIM_DEBUG_ERROR, "perl", | |
443 "Perl function %s exited abnormally: %s\n", | |
444 gps->load_sub, SvPV(ERRSV, len)); | |
445 } | |
446 | |
447 PUTBACK; | |
448 FREETMPS; | |
449 LEAVE; | |
450 } | |
451 | |
452 return TRUE; | |
453 } | |
454 | |
455 static void | |
456 destroy_package(const char *package) | |
457 { | |
458 dSP; | |
459 PERL_SET_CONTEXT(my_perl); | |
460 SPAGAIN; | |
461 | |
462 ENTER; | |
463 SAVETMPS; | |
464 | |
465 PUSHMARK(SP); | |
466 XPUSHs(sv_2mortal(newSVpv(package, strlen(package)))); | |
467 PUTBACK; | |
468 | |
469 perl_call_pv("Gaim::PerlLoader::destroy_package", | |
470 G_VOID | G_EVAL | G_DISCARD); | |
471 | |
472 SPAGAIN; | |
473 | |
474 PUTBACK; | |
475 FREETMPS; | |
476 LEAVE; | |
477 } | |
478 | |
479 static gboolean | |
480 unload_perl_plugin(GaimPlugin *plugin) | |
481 { | |
482 GaimPerlScript *gps = (GaimPerlScript *)plugin->info->extra_info; | |
483 | |
484 if (gps == NULL) | |
485 return FALSE; | |
486 | |
487 gaim_debug(GAIM_DEBUG_INFO, "perl", "Unloading perl script\n"); | |
488 | |
489 if (gps->unload_sub != NULL) { | |
490 dSP; | |
491 PERL_SET_CONTEXT(my_perl); | |
492 SPAGAIN; | |
493 ENTER; | |
494 SAVETMPS; | |
495 PUSHMARK(sp); | |
496 XPUSHs(sv_2mortal(gaim_perl_bless_object(plugin, | |
497 "Gaim::Plugin"))); | |
498 PUTBACK; | |
499 | |
500 perl_call_pv(gps->unload_sub, G_EVAL | G_SCALAR); | |
501 SPAGAIN; | |
502 | |
503 if (SvTRUE(ERRSV)) { | |
504 STRLEN len; | |
505 | |
506 gaim_debug(GAIM_DEBUG_ERROR, "perl", | |
507 "Perl function %s exited abnormally: %s\n", | |
508 gps->load_sub, SvPV(ERRSV, len)); | |
509 } | |
510 | |
511 PUTBACK; | |
512 FREETMPS; | |
513 LEAVE; | |
514 } | |
515 | |
516 gaim_perl_cmd_clear_for_plugin(plugin); | |
517 gaim_perl_signal_clear_for_plugin(plugin); | |
518 gaim_perl_timeout_clear_for_plugin(plugin); | |
519 | |
520 destroy_package(gps->package); | |
521 | |
522 return TRUE; | |
523 } | |
524 | |
525 static void | |
526 destroy_perl_plugin(GaimPlugin *plugin) | |
527 { | |
528 if (plugin->info != NULL) { | |
529 GaimPerlScript *gps; | |
530 | |
531 g_free(plugin->info->name); | |
532 g_free(plugin->info->version); | |
533 g_free(plugin->info->summary); | |
534 g_free(plugin->info->description); | |
535 g_free(plugin->info->author); | |
536 g_free(plugin->info->homepage); | |
537 | |
538 gps = (GaimPerlScript *)plugin->info->extra_info; | |
539 if (gps != NULL) { | |
540 g_free(gps->load_sub); | |
541 g_free(gps->unload_sub); | |
542 g_free(gps->package); | |
543 g_free(gps->prefs_sub); | |
14364
338ac096e322
[gaim-migrate @ 17070]
Daniel Atallah <daniel.atallah@gmail.com>
parents:
14192
diff
changeset
|
544 #ifdef GAIM_GTKPERL |
14192 | 545 g_free(gps->gtk_prefs_sub); |
14364
338ac096e322
[gaim-migrate @ 17070]
Daniel Atallah <daniel.atallah@gmail.com>
parents:
14192
diff
changeset
|
546 #endif |
14192 | 547 g_free(gps); |
548 plugin->info->extra_info = NULL; | |
549 } | |
550 } | |
551 } | |
552 | |
553 static gboolean | |
554 plugin_load(GaimPlugin *plugin) | |
555 { | |
556 return TRUE; | |
557 } | |
558 | |
559 static gboolean | |
560 plugin_unload(GaimPlugin *plugin) | |
561 { | |
562 perl_end(); | |
563 | |
564 return TRUE; | |
565 } | |
566 | |
567 static GaimPluginLoaderInfo loader_info = | |
568 { | |
569 NULL, /**< exts */ | |
570 probe_perl_plugin, /**< probe */ | |
571 load_perl_plugin, /**< load */ | |
572 unload_perl_plugin, /**< unload */ | |
573 destroy_perl_plugin /**< destroy */ | |
574 }; | |
575 | |
576 static GaimPluginInfo info = | |
577 { | |
578 GAIM_PLUGIN_MAGIC, | |
579 GAIM_MAJOR_VERSION, | |
580 GAIM_MINOR_VERSION, | |
581 GAIM_PLUGIN_LOADER, /**< type */ | |
582 NULL, /**< ui_requirement */ | |
583 0, /**< flags */ | |
584 NULL, /**< dependencies */ | |
585 GAIM_PRIORITY_DEFAULT, /**< priority */ | |
586 | |
587 PERL_PLUGIN_ID, /**< id */ | |
588 N_("Perl Plugin Loader"), /**< name */ | |
589 VERSION, /**< version */ | |
590 N_("Provides support for loading perl plugins."), /**< summary */ | |
591 N_("Provides support for loading perl plugins."), /**< description */ | |
592 "Christian Hammond <chipx86@gnupdate.org>", /**< author */ | |
593 GAIM_WEBSITE, /**< homepage */ | |
594 | |
595 plugin_load, /**< load */ | |
596 plugin_unload, /**< unload */ | |
597 NULL, /**< destroy */ | |
598 | |
599 NULL, /**< ui_info */ | |
600 &loader_info, /**< extra_info */ | |
601 NULL, | |
602 NULL | |
603 }; | |
604 | |
605 static void | |
606 init_plugin(GaimPlugin *plugin) | |
607 { | |
608 loader_info.exts = g_list_append(loader_info.exts, "pl"); | |
609 } | |
610 | |
611 GAIM_INIT_PLUGIN(perl, init_plugin, info) |