Debug: internal consistency under testsuite
[exim.git] / src / src / perl.c
CommitLineData
059ec3d9
PH
1/*************************************************
2* Exim - an Internet mail transport agent *
3*************************************************/
4
5/* Copyright (c) 1998 Malcolm Beattie */
f9ba5e22 6/* Copyright (C) 1999 - 2018 Exim maintainers */
059ec3d9
PH
7
8/* Modified by PH to get rid of the "na" usage, March 1999.
9 Modified further by PH for general tidying for Exim 4.
10 Threaded Perl support added by Stefan Traby, Nov 2002
11*/
12
13
14/* This Perl add-on can be distributed under the same terms as Exim itself. */
15/* See the file NOTICE for conditions of use and distribution. */
16
2f680c0c 17#include <assert.h>
059ec3d9
PH
18#include "exim.h"
19
20#define EXIM_TRUE TRUE
21#undef TRUE
22
23#define EXIM_FALSE FALSE
24#undef FALSE
25
26#define EXIM_DEBUG DEBUG
27#undef DEBUG
28
29#include <EXTERN.h>
30#include <perl.h>
31#include <XSUB.h>
32
33#ifndef ERRSV
34#define ERRSV (GvSV(errgv))
35#endif
36
37/* Some people like very old perl versions, so avoid any build side-effects. */
38
39#ifndef pTHX
40# define pTHX
41# define pTHX_
42#endif
43#ifndef EXTERN_C
44# define EXTERN_C extern
45#endif
46
47EXTERN_C void boot_DynaLoader(pTHX_ CV *cv);
48
49
50static PerlInterpreter *interp_perl = 0;
51
52XS(xs_expand_string)
53{
54 dXSARGS;
55 uschar *str;
56 STRLEN len;
57
58 if (items != 1)
59 croak("Usage: Exim::expand_string(string)");
60
61 str = expand_string(US SvPV(ST(0), len));
62 ST(0) = sv_newmortal();
63 if (str != NULL)
5903c6ff 64 sv_setpv(ST(0), CCS str);
8768d548 65 else if (!f.expand_string_forcedfail)
059ec3d9
PH
66 croak("syntax error in Exim::expand_string argument: %s",
67 expand_string_message);
68}
69
70XS(xs_debug_write)
71{
72 dXSARGS;
73 STRLEN len;
74 if (items != 1)
75 croak("Usage: Exim::debug_write(string)");
76 debug_printf("%s", US SvPV(ST(0), len));
77}
78
79XS(xs_log_write)
80{
81 dXSARGS;
82 STRLEN len;
83 if (items != 1)
84 croak("Usage: Exim::log_write(string)");
85 log_write(0, LOG_MAIN, "%s", US SvPV(ST(0), len));
86}
87
88static void xs_init(pTHX)
89{
90 char *file = __FILE__;
91 newXS("DynaLoader::boot_DynaLoader", boot_DynaLoader, file);
92 newXS("Exim::expand_string", xs_expand_string, file);
93 newXS("Exim::debug_write", xs_debug_write, file);
94 newXS("Exim::log_write", xs_log_write, file);
95}
96
97uschar *
98init_perl(uschar *startup_code)
99{
2f680c0c
HSHR
100 static int argc = 1;
101 static char *argv[4] = { "exim-perl" };
059ec3d9
PH
102 SV *sv;
103 STRLEN len;
104
2f680c0c
HSHR
105 if (opt_perl_taintmode) argv[argc++] = "-T";
106 argv[argc++] = "/dev/null";
107 argv[argc] = 0;
108
109 assert(sizeof(argv)/sizeof(argv[0]) > argc);
110
059ec3d9
PH
111 if (interp_perl) return 0;
112 interp_perl = perl_alloc();
113 perl_construct(interp_perl);
114 perl_parse(interp_perl, xs_init, argc, argv, 0);
115 perl_run(interp_perl);
116 {
117 dSP;
76a2d7ba
PH
118
119 /*********************************************************************/
120 /* These lines by PH added to make "warn" output go to the Exim log; I
121 hope this doesn't break anything. */
8e669ac1 122
76a2d7ba
PH
123 sv = newSVpv(
124 "$SIG{__WARN__} = sub { my($s) = $_[0];"
8e669ac1 125 "$s =~ s/\\n$//;"
76a2d7ba
PH
126 "Exim::log_write($s) };", 0);
127 PUSHMARK(SP);
128 perl_eval_sv(sv, G_SCALAR|G_DISCARD|G_KEEPERR);
129 SvREFCNT_dec(sv);
130 if (SvTRUE(ERRSV)) return US SvPV(ERRSV, len);
131 /*********************************************************************/
8e669ac1 132
059ec3d9
PH
133 sv = newSVpv(CS startup_code, 0);
134 PUSHMARK(SP);
135 perl_eval_sv(sv, G_SCALAR|G_DISCARD|G_KEEPERR);
136 SvREFCNT_dec(sv);
137 if (SvTRUE(ERRSV)) return US SvPV(ERRSV, len);
35d40a98
PH
138
139 setlocale(LC_ALL, "C"); /* In case it got changed */
059ec3d9
PH
140 return NULL;
141 }
142}
143
144void
145cleanup_perl(void)
146{
147 if (!interp_perl)
148 return;
149 perl_destruct(interp_perl);
150 perl_free(interp_perl);
151 interp_perl = 0;
152}
153
acec9514
JH
154gstring *
155call_perl_cat(gstring * yield, uschar **errstrp, uschar *name, uschar **arg)
059ec3d9
PH
156{
157 dSP;
158 SV *sv;
159 STRLEN len;
160 uschar *str;
161 int items;
162
163 if (!interp_perl)
164 {
165 *errstrp = US"the Perl interpreter has not been started";
166 return 0;
167 }
168
169 ENTER;
170 SAVETMPS;
171 PUSHMARK(SP);
172 while (*arg != NULL) XPUSHs(newSVpv(CS (*arg++), 0));
173 PUTBACK;
174 items = perl_call_pv(CS name, G_SCALAR|G_EVAL);
175 SPAGAIN;
176 sv = POPs;
177 PUTBACK;
178 if (SvTRUE(ERRSV))
179 {
180 *errstrp = US SvPV(ERRSV, len);
181 return NULL;
182 }
183 if (!SvOK(sv))
184 {
185 *errstrp = 0;
186 return NULL;
187 }
188 str = US SvPV(sv, len);
acec9514 189 yield = string_catn(yield, str, (int)len);
059ec3d9
PH
190 FREETMPS;
191 LEAVE;
8e669ac1 192
a444213a 193 setlocale(LC_ALL, "C"); /* In case it got changed */
059ec3d9
PH
194 return yield;
195}
196
197/* End of perl.c */