mirror of
https://github.com/postgres/postgres.git
synced 2025-04-25 21:42:33 +03:00
New versions of perl trigger warnings within perl.h with our compiler flags. At least -Wdeclaration-after-statement, -Wshadow=compatible-local are known to be problematic. To avoid these warnings, conditionally use #pragma GCC system_header before including plperl.h. Alternatively, we could add the include paths for problematic headers with -isystem, but that is a larger hammer and is harder to search for. A more granular alternative would be to use #pragma GCC diagnostic push/ignored/pop, but gcc warns about unknown warnings being ignored, so every to-be-ignored-temporarily compiler warning would require its own pg_config.h symbol and #ifdef. As the warnings are voluminous, it makes sense to backpatch this change. But don't do so yet, we first want gather buildfarm coverage - it's e.g. possible that some compiler claiming to be gcc compatible has issues with the pragma. Author: Andres Freund <andres@anarazel.de> Reviewed-by: Tom Lane <tgl@sss.pgh.pa.us> Discussion: Discussion: https://postgr.es/m/20221228182455.hfdwd22zztvkojy2@awork3.anarazel.de
396 lines
9.6 KiB
C
396 lines
9.6 KiB
C
/*-------------------------------------------------------------------------
|
|
*
|
|
* plperl.h
|
|
* Common include file for PL/Perl files
|
|
*
|
|
* This should be included _AFTER_ postgres.h and system include files, as
|
|
* well as headers that could in turn include system headers.
|
|
*
|
|
* Portions Copyright (c) 1996-2022, PostgreSQL Global Development Group
|
|
* Portions Copyright (c) 1995, Regents of the University of California
|
|
*
|
|
* src/pl/plperl/plperl.h
|
|
*/
|
|
|
|
#ifndef PL_PERL_H
|
|
#define PL_PERL_H
|
|
|
|
/* defines free() by way of system headers, so must be included before perl.h */
|
|
#include "mb/pg_wchar.h"
|
|
|
|
/* stop perl headers from hijacking stdio and other stuff on Windows */
|
|
#ifdef WIN32
|
|
#define WIN32IO_IS_STDIO
|
|
#endif /* WIN32 */
|
|
|
|
/*
|
|
* Supply a value of PERL_UNUSED_DECL that will satisfy gcc - the one
|
|
* perl itself supplies doesn't seem to.
|
|
*/
|
|
#define PERL_UNUSED_DECL pg_attribute_unused()
|
|
|
|
/*
|
|
* Sometimes perl carefully scribbles on our *printf macros.
|
|
* So we undefine them here and redefine them after it's done its dirty deed.
|
|
*/
|
|
#undef vsnprintf
|
|
#undef snprintf
|
|
#undef vsprintf
|
|
#undef sprintf
|
|
#undef vfprintf
|
|
#undef fprintf
|
|
#undef vprintf
|
|
#undef printf
|
|
|
|
/*
|
|
* Perl scribbles on the "_" macro too.
|
|
*/
|
|
#undef _
|
|
|
|
/*
|
|
* ActivePerl 5.18 and later are MinGW-built, and their headers use GCC's
|
|
* __inline__. Translate to something MSVC recognizes. Also, perl.h sometimes
|
|
* defines isnan, so undefine it here and put back the definition later if
|
|
* perl.h doesn't.
|
|
*/
|
|
#ifdef _MSC_VER
|
|
#define __inline__ inline
|
|
#ifdef isnan
|
|
#undef isnan
|
|
#endif
|
|
/* Work around for using MSVC and Strawberry Perl >= 5.30. */
|
|
#define __builtin_expect(expr, val) (expr)
|
|
#endif
|
|
|
|
/*
|
|
* Regarding bool, both PostgreSQL and Perl might use stdbool.h or not,
|
|
* depending on configuration. If both agree, things are relatively harmless.
|
|
* If not, things get tricky. If PostgreSQL does but Perl does not, define
|
|
* HAS_BOOL here so that Perl does not redefine bool; this avoids compiler
|
|
* warnings. If PostgreSQL does not but Perl does, we need to undefine bool
|
|
* after we include the Perl headers; see below.
|
|
*/
|
|
#ifdef PG_USE_STDBOOL
|
|
#define HAS_BOOL 1
|
|
#endif
|
|
|
|
/*
|
|
* Newer versions of the perl headers trigger a lot of warnings with our
|
|
* compiler flags (at least -Wdeclaration-after-statement,
|
|
* -Wshadow=compatible-local are known to be problematic). The system_header
|
|
* pragma hides warnings from within the rest of this file, if supported.
|
|
*/
|
|
#ifdef HAVE_PRAGMA_GCC_SYSTEM_HEADER
|
|
#pragma GCC system_header
|
|
#endif
|
|
|
|
/*
|
|
* Get the basic Perl API. We use PERL_NO_GET_CONTEXT mode so that our code
|
|
* can compile against MULTIPLICITY Perl builds without including XSUB.h.
|
|
*/
|
|
#define PERL_NO_GET_CONTEXT
|
|
#include "EXTERN.h"
|
|
#include "perl.h"
|
|
|
|
/*
|
|
* We want to include XSUB.h only within .xs files, because on some platforms
|
|
* it undesirably redefines a lot of libc functions. But it must appear
|
|
* before ppport.h, so use a #define flag to control inclusion here.
|
|
*/
|
|
#ifdef PG_NEED_PERL_XSUB_H
|
|
/*
|
|
* On Windows, win32_port.h defines macros for a lot of these same functions.
|
|
* To avoid compiler warnings when XSUB.h redefines them, #undef our versions.
|
|
*/
|
|
#ifdef WIN32
|
|
#undef accept
|
|
#undef bind
|
|
#undef connect
|
|
#undef fopen
|
|
#undef fstat
|
|
#undef kill
|
|
#undef listen
|
|
#undef lstat
|
|
#undef mkdir
|
|
#undef open
|
|
#undef putenv
|
|
#undef recv
|
|
#undef rename
|
|
#undef select
|
|
#undef send
|
|
#undef socket
|
|
#undef stat
|
|
#undef unlink
|
|
#endif
|
|
|
|
#include "XSUB.h"
|
|
#endif
|
|
|
|
/* put back our *printf macros ... this must match src/include/port.h */
|
|
#ifdef vsnprintf
|
|
#undef vsnprintf
|
|
#endif
|
|
#ifdef snprintf
|
|
#undef snprintf
|
|
#endif
|
|
#ifdef vsprintf
|
|
#undef vsprintf
|
|
#endif
|
|
#ifdef sprintf
|
|
#undef sprintf
|
|
#endif
|
|
#ifdef vfprintf
|
|
#undef vfprintf
|
|
#endif
|
|
#ifdef fprintf
|
|
#undef fprintf
|
|
#endif
|
|
#ifdef vprintf
|
|
#undef vprintf
|
|
#endif
|
|
#ifdef printf
|
|
#undef printf
|
|
#endif
|
|
|
|
#define vsnprintf pg_vsnprintf
|
|
#define snprintf pg_snprintf
|
|
#define vsprintf pg_vsprintf
|
|
#define sprintf pg_sprintf
|
|
#define vfprintf pg_vfprintf
|
|
#define fprintf pg_fprintf
|
|
#define vprintf pg_vprintf
|
|
#define printf(...) pg_printf(__VA_ARGS__)
|
|
|
|
/*
|
|
* Put back "_" too; but rather than making it just gettext() as the core
|
|
* code does, make it dgettext() so that the right things will happen in
|
|
* loadable modules (if they've set up TEXTDOMAIN correctly). Note that
|
|
* we can't just set TEXTDOMAIN here, because this file is used by more
|
|
* extensions than just PL/Perl itself.
|
|
*/
|
|
#undef _
|
|
#define _(x) dgettext(TEXTDOMAIN, x)
|
|
|
|
/* put back the definition of isnan if needed */
|
|
#ifdef _MSC_VER
|
|
#ifndef isnan
|
|
#define isnan(x) _isnan(x)
|
|
#endif
|
|
#endif
|
|
|
|
/* perl version and platform portability */
|
|
#include "ppport.h"
|
|
|
|
/*
|
|
* perl might have included stdbool.h. If we also did that earlier (see c.h),
|
|
* then that's fine. If not, we probably rejected it for some reason. In
|
|
* that case, undef bool and proceed with our own bool. (Note that stdbool.h
|
|
* makes bool a macro, but our own replacement is a typedef, so the undef
|
|
* makes ours visible again).
|
|
*/
|
|
#ifndef PG_USE_STDBOOL
|
|
#ifdef bool
|
|
#undef bool
|
|
#endif
|
|
#endif
|
|
|
|
/* supply HeUTF8 if it's missing - ppport.h doesn't supply it, unfortunately */
|
|
#ifndef HeUTF8
|
|
#define HeUTF8(he) ((HeKLEN(he) == HEf_SVKEY) ? \
|
|
SvUTF8(HeKEY_sv(he)) : \
|
|
(U32)HeKUTF8(he))
|
|
#endif
|
|
|
|
/* supply GvCV_set if it's missing - ppport.h doesn't supply it, unfortunately */
|
|
#ifndef GvCV_set
|
|
#define GvCV_set(gv, cv) (GvCV(gv) = cv)
|
|
#endif
|
|
|
|
/* Perl 5.19.4 changed array indices from I32 to SSize_t */
|
|
#if PERL_BCDVERSION >= 0x5019004
|
|
#define AV_SIZE_MAX SSize_t_MAX
|
|
#else
|
|
#define AV_SIZE_MAX I32_MAX
|
|
#endif
|
|
|
|
/* declare routines from plperl.c for access by .xs files */
|
|
HV *plperl_spi_exec(char *, int);
|
|
void plperl_return_next(SV *);
|
|
SV *plperl_spi_query(char *);
|
|
SV *plperl_spi_fetchrow(char *);
|
|
SV *plperl_spi_prepare(char *, int, SV **);
|
|
HV *plperl_spi_exec_prepared(char *, HV *, int, SV **);
|
|
SV *plperl_spi_query_prepared(char *, int, SV **);
|
|
void plperl_spi_freeplan(char *);
|
|
void plperl_spi_cursor_close(char *);
|
|
void plperl_spi_commit(void);
|
|
void plperl_spi_rollback(void);
|
|
char *plperl_sv_to_literal(SV *, char *);
|
|
void plperl_util_elog(int level, SV *msg);
|
|
|
|
|
|
/* helper functions */
|
|
|
|
/*
|
|
* convert from utf8 to database encoding
|
|
*
|
|
* Returns a palloc'ed copy of the original string
|
|
*/
|
|
static inline char *
|
|
utf_u2e(char *utf8_str, size_t len)
|
|
{
|
|
char *ret;
|
|
|
|
ret = pg_any_to_server(utf8_str, len, PG_UTF8);
|
|
|
|
/* ensure we have a copy even if no conversion happened */
|
|
if (ret == utf8_str)
|
|
ret = pstrdup(ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* convert from database encoding to utf8
|
|
*
|
|
* Returns a palloc'ed copy of the original string
|
|
*/
|
|
static inline char *
|
|
utf_e2u(const char *str)
|
|
{
|
|
char *ret;
|
|
|
|
ret = pg_server_to_any(str, strlen(str), PG_UTF8);
|
|
|
|
/* ensure we have a copy even if no conversion happened */
|
|
if (ret == str)
|
|
ret = pstrdup(ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Convert an SV to a char * in the current database encoding
|
|
*
|
|
* Returns a palloc'ed copy of the original string
|
|
*/
|
|
static inline char *
|
|
sv2cstr(SV *sv)
|
|
{
|
|
dTHX;
|
|
char *val,
|
|
*res;
|
|
STRLEN len;
|
|
|
|
/*
|
|
* get a utf8 encoded char * out of perl. *note* it may not be valid utf8!
|
|
*/
|
|
|
|
/*
|
|
* SvPVutf8() croaks nastily on certain things, like typeglobs and
|
|
* readonly objects such as $^V. That's a perl bug - it's not supposed to
|
|
* happen. To avoid crashing the backend, we make a copy of the sv before
|
|
* passing it to SvPVutf8(). The copy is garbage collected when we're done
|
|
* with it.
|
|
*/
|
|
if (SvREADONLY(sv) ||
|
|
isGV_with_GP(sv) ||
|
|
(SvTYPE(sv) > SVt_PVLV && SvTYPE(sv) != SVt_PVFM))
|
|
sv = newSVsv(sv);
|
|
else
|
|
{
|
|
/*
|
|
* increase the reference count so we can just SvREFCNT_dec() it when
|
|
* we are done
|
|
*/
|
|
SvREFCNT_inc_simple_void(sv);
|
|
}
|
|
|
|
/*
|
|
* Request the string from Perl, in UTF-8 encoding; but if we're in a
|
|
* SQL_ASCII database, just request the byte soup without trying to make
|
|
* it UTF8, because that might fail.
|
|
*/
|
|
if (GetDatabaseEncoding() == PG_SQL_ASCII)
|
|
val = SvPV(sv, len);
|
|
else
|
|
val = SvPVutf8(sv, len);
|
|
|
|
/*
|
|
* Now convert to database encoding. We use perl's length in the event we
|
|
* had an embedded null byte to ensure we error out properly.
|
|
*/
|
|
res = utf_u2e(val, len);
|
|
|
|
/* safe now to garbage collect the new SV */
|
|
SvREFCNT_dec(sv);
|
|
|
|
return res;
|
|
}
|
|
|
|
/*
|
|
* Create a new SV from a string assumed to be in the current database's
|
|
* encoding.
|
|
*/
|
|
static inline SV *
|
|
cstr2sv(const char *str)
|
|
{
|
|
dTHX;
|
|
SV *sv;
|
|
char *utf8_str;
|
|
|
|
/* no conversion when SQL_ASCII */
|
|
if (GetDatabaseEncoding() == PG_SQL_ASCII)
|
|
return newSVpv(str, 0);
|
|
|
|
utf8_str = utf_e2u(str);
|
|
|
|
sv = newSVpv(utf8_str, 0);
|
|
SvUTF8_on(sv);
|
|
pfree(utf8_str);
|
|
|
|
return sv;
|
|
}
|
|
|
|
/*
|
|
* croak() with specified message, which is given in the database encoding.
|
|
*
|
|
* Ideally we'd just write croak("%s", str), but plain croak() does not play
|
|
* nice with non-ASCII data. In modern Perl versions we can call cstr2sv()
|
|
* and pass the result to croak_sv(); in versions that don't have croak_sv(),
|
|
* we have to work harder.
|
|
*/
|
|
static inline void
|
|
croak_cstr(const char *str)
|
|
{
|
|
dTHX;
|
|
|
|
#ifdef croak_sv
|
|
/* Use sv_2mortal() to be sure the transient SV gets freed */
|
|
croak_sv(sv_2mortal(cstr2sv(str)));
|
|
#else
|
|
|
|
/*
|
|
* The older way to do this is to assign a UTF8-marked value to ERRSV and
|
|
* then call croak(NULL). But if we leave it to croak() to append the
|
|
* error location, it does so too late (only after popping the stack) in
|
|
* some Perl versions. Hence, use mess() to create an SV with the error
|
|
* location info already appended.
|
|
*/
|
|
SV *errsv = get_sv("@", GV_ADD);
|
|
char *utf8_str = utf_e2u(str);
|
|
SV *ssv;
|
|
|
|
ssv = mess("%s", utf8_str);
|
|
SvUTF8_on(ssv);
|
|
|
|
pfree(utf8_str);
|
|
|
|
sv_setsv(errsv, ssv);
|
|
|
|
croak(NULL);
|
|
#endif /* croak_sv */
|
|
}
|
|
|
|
#endif /* PL_PERL_H */
|