1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172
|
/***********************************************************************
*
* embperl.c
*
* Routines for manipulating embedded Perl interpreter
*
* Copyright (C) 2003 by Roaring Penguin Software Inc.
*
* This program may be distributed under the terms of the GNU General
* Public License, Version 2.
*
***********************************************************************/
#ifdef EMBED_PERL
#include <EXTERN.h>
#include <perl.h>
#include <errno.h>
#include <syslog.h>
#ifdef PERL_SET_CONTEXT
#define PSC(x) PERL_SET_CONTEXT(x)
#else
#define PSC(x) (void) 0
#endif
#define PERLPARSE_NUM_ARGS 6
static PerlInterpreter *my_perl = NULL;
extern void xs_init (pTHX);
void
init_embedded_interpreter(int argc, char **argv, char **env)
{
#ifdef PERL_SYS_INIT3
PERL_SYS_INIT3(&argc, &argv, &env);
#endif
}
void
deinit_embedded_interpreter(void)
{
#ifdef PERL_SYS_TERM
PERL_SYS_TERM();
#endif
}
void
term_embedded_interpreter(void)
{
if (my_perl != NULL) {
PSC(my_perl);
PERL_SET_INTERP(my_perl);
PL_perl_destruct_level = 1;
perl_destruct(my_perl);
perl_free(my_perl);
my_perl = NULL;
}
}
static char **argv = NULL;
int
make_embedded_interpreter(char const *progPath,
char const *subFilter,
int wantStatusReports,
char **env)
{
int argc;
/* Why do we malloc argv instead of making it static? Because on some
systems, Perl makes horrendously evil assumptions about the alignment
of argv... we use malloc to get guaranteed worst-case alignment.
Yes, the Perl innards are completely horrible. */
if (!argv) {
argv = (char **) malloc(PERLPARSE_NUM_ARGS * sizeof(char *));
if (!argv) {
fprintf(stderr, "Out of memory allocating argv[] array for embedded Perl!");
syslog(LOG_ERR, "Out of memory allocating argv[] array for embedded Perl!");
exit(EXIT_FAILURE);
}
}
memset(argv, 0, PERLPARSE_NUM_ARGS * sizeof(char *));
if (my_perl != NULL) {
#ifdef SAFE_EMBED_PERL
PSC(my_perl);
PERL_SET_INTERP(my_perl);
PL_perl_destruct_level = 1;
perl_destruct(my_perl);
perl_free(my_perl);
my_perl = NULL;
#ifdef PERL_SET_CONTEXT
PERL_SET_CONTEXT(NULL);
#endif
/* The next line should NOT be necessary, but perl 5.10 appears to have
a bug as described at http://bugs.debian.org/cgi-bin/bugreport.cgi?bug=516913
*/
#ifdef PERL_SET_INTERP
PERL_SET_INTERP(NULL);
#endif
#else
syslog(LOG_WARNING, "Cannot destroy and recreate a Perl interpreter safely on this platform. Filter rules will NOT be reread.");
return 0;
#endif
}
if (subFilter) {
argv[0] = "";
argv[1] = (char *) progPath;
argv[2] = "-f";
argv[3] = (char *) subFilter;
if (wantStatusReports) {
argv[4] = "-embserveru";
} else {
argv[4] = "-embserver";
}
argv[5] = NULL;
argc = 5;
} else {
argv[0] = "";
argv[1] = (char *) progPath;
if (wantStatusReports) {
argv[2] = "-embserveru";
} else {
argv[2] = "-embserver";
}
argv[3] = NULL;
argc = 3;
}
my_perl = perl_alloc();
if (!my_perl) {
errno = ENOMEM;
return -1;
}
PSC(my_perl);
PERL_SET_INTERP(my_perl);
PL_perl_destruct_level = 1;
perl_construct(my_perl);
perl_parse(my_perl, xs_init, argc, argv, NULL);
perl_run(my_perl);
return 0;
}
/* Perl caches $$ so the PID is wrong after we fork. This
routine fixes it up */
static void
embperl_fix_pid(void)
{
GV *tmpgv;
if ((tmpgv = gv_fetchpv("$",TRUE, SVt_PV))) {
SvREADONLY_off(GvSV(tmpgv));
sv_setiv(GvSV(tmpgv), PerlProc_getpid());
SvREADONLY_on(GvSV(tmpgv));
}
}
void
run_embedded_filter(void)
{
char *args[] = { NULL };
PSC(my_perl);
PERL_SET_INTERP(my_perl);
embperl_fix_pid();
perl_call_argv("do_main_loop", G_DISCARD | G_NOARGS, args);
}
#endif
|