2015-04-26 22:33:14 +08:00
|
|
|
#include "postgres.h"
|
2016-10-05 05:49:07 +08:00
|
|
|
|
2015-04-26 22:33:14 +08:00
|
|
|
#undef _
|
2016-10-05 05:49:07 +08:00
|
|
|
|
2015-04-26 22:33:14 +08:00
|
|
|
#include "fmgr.h"
|
|
|
|
#include "plperl.h"
|
|
|
|
#include "plperl_helpers.h"
|
2018-09-17 01:46:45 +08:00
|
|
|
#include "hstore/hstore.h"
|
2015-04-26 22:33:14 +08:00
|
|
|
|
|
|
|
PG_MODULE_MAGIC;
|
|
|
|
|
2016-10-05 05:49:07 +08:00
|
|
|
extern void _PG_init(void);
|
|
|
|
|
|
|
|
/* Linkage to functions in hstore module */
|
|
|
|
typedef HStore *(*hstoreUpgrade_t) (Datum orig);
|
|
|
|
static hstoreUpgrade_t hstoreUpgrade_p;
|
|
|
|
typedef int (*hstoreUniquePairs_t) (Pairs *a, int32 l, int32 *buflen);
|
|
|
|
static hstoreUniquePairs_t hstoreUniquePairs_p;
|
|
|
|
typedef HStore *(*hstorePairs_t) (Pairs *pairs, int32 pcount, int32 buflen);
|
|
|
|
static hstorePairs_t hstorePairs_p;
|
|
|
|
typedef size_t (*hstoreCheckKeyLen_t) (size_t len);
|
|
|
|
static hstoreCheckKeyLen_t hstoreCheckKeyLen_p;
|
|
|
|
typedef size_t (*hstoreCheckValLen_t) (size_t len);
|
|
|
|
static hstoreCheckValLen_t hstoreCheckValLen_p;
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Module initialize function: fetch function pointers for cross-module calls.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
_PG_init(void)
|
|
|
|
{
|
|
|
|
/* Asserts verify that typedefs above match original declarations */
|
|
|
|
AssertVariableIsOfType(&hstoreUpgrade, hstoreUpgrade_t);
|
|
|
|
hstoreUpgrade_p = (hstoreUpgrade_t)
|
|
|
|
load_external_function("$libdir/hstore", "hstoreUpgrade",
|
|
|
|
true, NULL);
|
|
|
|
AssertVariableIsOfType(&hstoreUniquePairs, hstoreUniquePairs_t);
|
|
|
|
hstoreUniquePairs_p = (hstoreUniquePairs_t)
|
|
|
|
load_external_function("$libdir/hstore", "hstoreUniquePairs",
|
|
|
|
true, NULL);
|
|
|
|
AssertVariableIsOfType(&hstorePairs, hstorePairs_t);
|
|
|
|
hstorePairs_p = (hstorePairs_t)
|
|
|
|
load_external_function("$libdir/hstore", "hstorePairs",
|
|
|
|
true, NULL);
|
|
|
|
AssertVariableIsOfType(&hstoreCheckKeyLen, hstoreCheckKeyLen_t);
|
|
|
|
hstoreCheckKeyLen_p = (hstoreCheckKeyLen_t)
|
|
|
|
load_external_function("$libdir/hstore", "hstoreCheckKeyLen",
|
|
|
|
true, NULL);
|
|
|
|
AssertVariableIsOfType(&hstoreCheckValLen, hstoreCheckValLen_t);
|
|
|
|
hstoreCheckValLen_p = (hstoreCheckValLen_t)
|
|
|
|
load_external_function("$libdir/hstore", "hstoreCheckValLen",
|
|
|
|
true, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* These defines must be after the module init function */
|
|
|
|
#define hstoreUpgrade hstoreUpgrade_p
|
|
|
|
#define hstoreUniquePairs hstoreUniquePairs_p
|
|
|
|
#define hstorePairs hstorePairs_p
|
|
|
|
#define hstoreCheckKeyLen hstoreCheckKeyLen_p
|
|
|
|
#define hstoreCheckValLen hstoreCheckValLen_p
|
|
|
|
|
2015-04-26 22:33:14 +08:00
|
|
|
|
|
|
|
PG_FUNCTION_INFO_V1(hstore_to_plperl);
|
|
|
|
|
|
|
|
Datum
|
|
|
|
hstore_to_plperl(PG_FUNCTION_ARGS)
|
|
|
|
{
|
PL/Perl portability fix: avoid including XSUB.h in plperl.c.
In Perl builds that define PERL_IMPLICIT_SYS, XSUB.h defines macros
that replace a whole lot of basic libc functions with Perl functions.
We can't tolerate that in plperl.c; it breaks at least PG_TRY and
probably other stuff. The core idea of this patch is to include XSUB.h
only in the .xs files where it's really needed, and to move any code
broken by PERL_IMPLICIT_SYS out of the .xs files and into plperl.c.
The reason this hasn't been a problem before is that our build techniques
did not result in PERL_IMPLICIT_SYS appearing as a #define in PL/Perl,
even on some platforms where Perl thinks it is defined. That's about to
change in order to fix a nasty portability issue, so we need this work to
make the code safe for that.
Rather unaccountably, the Perl people chose XSUB.h as the place to provide
the versions of the aTHX/aTHX_ macros that are needed by code that's not
explicitly aware of the MULTIPLICITY API conventions. Hence, just removing
XSUB.h from plperl.c fails miserably. But we can work around that by
defining PERL_NO_GET_CONTEXT (which would make the relevant stanza of
XSUB.h a no-op anyway). As explained in perlguts.pod, that means we need
to add a "dTHX" macro call in every C function that calls a Perl API
function. In most of them we just add this at the top; but since the
macro fetches the current Perl interpreter pointer, more care is needed
in functions that switch the active interpreter. Lack of the macro is
easily recognized since it results in bleats about "my_perl" not being
defined.
(A nice side benefit of this is that it significantly reduces the number
of fetches of the current interpreter pointer. On my machine, plperl.so
gets more than 10% smaller, and there's probably some performance win too.
We could reduce the number of fetches still more by decorating the code
with pTHX_/aTHX_ macros to pass the interpreter pointer around, as
explained by perlguts.pod; but that's a task for another day.)
Formatting note: pgindent seems happy to treat "dTHX;" as a declaration
so long as it's the first thing after the left brace, as we'd already
observed with respect to the similar macro "dSP;". If you try to put
it later in a set of declarations, pgindent puts ugly extra space
around it.
Having removed XSUB.h from plperl.c, we need only move the support
functions for spi_return_next and util_elog (both of which use PG_TRY)
out of the .xs files and into plperl.c. This seems sufficient to
avoid the known problems caused by PERL_IMPLICIT_SYS, although we
could move more code if additional issues emerge.
This will need to be back-patched, but first let's see what the
buildfarm makes of it.
Patch by me, with some help from Ashutosh Sharma
Discussion: https://postgr.es/m/CANFyU97OVQ3+Mzfmt3MhuUm5NwPU=-FtbNH5Eb7nZL9ua8=rcA@mail.gmail.com
2017-07-29 00:25:43 +08:00
|
|
|
dTHX;
|
2017-09-19 03:21:23 +08:00
|
|
|
HStore *in = PG_GETARG_HSTORE_P(0);
|
2015-04-26 22:33:14 +08:00
|
|
|
int i;
|
|
|
|
int count = HS_COUNT(in);
|
|
|
|
char *base = STRPTR(in);
|
|
|
|
HEntry *entries = ARRPTR(in);
|
|
|
|
HV *hv;
|
|
|
|
|
|
|
|
hv = newHV();
|
|
|
|
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
const char *key;
|
2015-05-24 09:35:49 +08:00
|
|
|
SV *value;
|
2015-04-26 22:33:14 +08:00
|
|
|
|
2015-11-20 03:54:05 +08:00
|
|
|
key = pnstrdup(HSTORE_KEY(entries, base, i),
|
|
|
|
HSTORE_KEYLEN(entries, i));
|
|
|
|
value = HSTORE_VALISNULL(entries, i) ? newSV(0) :
|
|
|
|
cstr2sv(pnstrdup(HSTORE_VAL(entries, base, i),
|
|
|
|
HSTORE_VALLEN(entries, i)));
|
2015-04-26 22:33:14 +08:00
|
|
|
|
|
|
|
(void) hv_store(hv, key, strlen(key), value, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return PointerGetDatum(newRV((SV *) hv));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
PG_FUNCTION_INFO_V1(plperl_to_hstore);
|
|
|
|
|
|
|
|
Datum
|
|
|
|
plperl_to_hstore(PG_FUNCTION_ARGS)
|
|
|
|
{
|
PL/Perl portability fix: avoid including XSUB.h in plperl.c.
In Perl builds that define PERL_IMPLICIT_SYS, XSUB.h defines macros
that replace a whole lot of basic libc functions with Perl functions.
We can't tolerate that in plperl.c; it breaks at least PG_TRY and
probably other stuff. The core idea of this patch is to include XSUB.h
only in the .xs files where it's really needed, and to move any code
broken by PERL_IMPLICIT_SYS out of the .xs files and into plperl.c.
The reason this hasn't been a problem before is that our build techniques
did not result in PERL_IMPLICIT_SYS appearing as a #define in PL/Perl,
even on some platforms where Perl thinks it is defined. That's about to
change in order to fix a nasty portability issue, so we need this work to
make the code safe for that.
Rather unaccountably, the Perl people chose XSUB.h as the place to provide
the versions of the aTHX/aTHX_ macros that are needed by code that's not
explicitly aware of the MULTIPLICITY API conventions. Hence, just removing
XSUB.h from plperl.c fails miserably. But we can work around that by
defining PERL_NO_GET_CONTEXT (which would make the relevant stanza of
XSUB.h a no-op anyway). As explained in perlguts.pod, that means we need
to add a "dTHX" macro call in every C function that calls a Perl API
function. In most of them we just add this at the top; but since the
macro fetches the current Perl interpreter pointer, more care is needed
in functions that switch the active interpreter. Lack of the macro is
easily recognized since it results in bleats about "my_perl" not being
defined.
(A nice side benefit of this is that it significantly reduces the number
of fetches of the current interpreter pointer. On my machine, plperl.so
gets more than 10% smaller, and there's probably some performance win too.
We could reduce the number of fetches still more by decorating the code
with pTHX_/aTHX_ macros to pass the interpreter pointer around, as
explained by perlguts.pod; but that's a task for another day.)
Formatting note: pgindent seems happy to treat "dTHX;" as a declaration
so long as it's the first thing after the left brace, as we'd already
observed with respect to the similar macro "dSP;". If you try to put
it later in a set of declarations, pgindent puts ugly extra space
around it.
Having removed XSUB.h from plperl.c, we need only move the support
functions for spi_return_next and util_elog (both of which use PG_TRY)
out of the .xs files and into plperl.c. This seems sufficient to
avoid the known problems caused by PERL_IMPLICIT_SYS, although we
could move more code if additional issues emerge.
This will need to be back-patched, but first let's see what the
buildfarm makes of it.
Patch by me, with some help from Ashutosh Sharma
Discussion: https://postgr.es/m/CANFyU97OVQ3+Mzfmt3MhuUm5NwPU=-FtbNH5Eb7nZL9ua8=rcA@mail.gmail.com
2017-07-29 00:25:43 +08:00
|
|
|
dTHX;
|
2018-06-19 03:55:06 +08:00
|
|
|
SV *in = (SV *) PG_GETARG_POINTER(0);
|
|
|
|
HV *hv;
|
2015-04-26 22:33:14 +08:00
|
|
|
HE *he;
|
|
|
|
int32 buflen;
|
|
|
|
int32 i;
|
|
|
|
int32 pcount;
|
|
|
|
HStore *out;
|
|
|
|
Pairs *pairs;
|
|
|
|
|
2018-06-19 03:55:06 +08:00
|
|
|
/* Dereference references recursively. */
|
|
|
|
while (SvROK(in))
|
|
|
|
in = SvRV(in);
|
|
|
|
|
|
|
|
/* Now we must have a hash. */
|
|
|
|
if (SvTYPE(in) != SVt_PVHV)
|
|
|
|
ereport(ERROR,
|
|
|
|
(errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
|
|
|
|
(errmsg("cannot transform non-hash Perl value to hstore"))));
|
|
|
|
hv = (HV *) in;
|
|
|
|
|
2015-04-26 22:33:14 +08:00
|
|
|
pcount = hv_iterinit(hv);
|
|
|
|
|
|
|
|
pairs = palloc(pcount * sizeof(Pairs));
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
while ((he = hv_iternext(hv)))
|
|
|
|
{
|
2015-05-24 09:35:49 +08:00
|
|
|
char *key = sv2cstr(HeSVKEY_force(he));
|
|
|
|
SV *value = HeVAL(he);
|
2015-04-26 22:33:14 +08:00
|
|
|
|
|
|
|
pairs[i].key = pstrdup(key);
|
|
|
|
pairs[i].keylen = hstoreCheckKeyLen(strlen(pairs[i].key));
|
|
|
|
pairs[i].needfree = true;
|
|
|
|
|
|
|
|
if (!SvOK(value))
|
|
|
|
{
|
|
|
|
pairs[i].val = NULL;
|
|
|
|
pairs[i].vallen = 0;
|
|
|
|
pairs[i].isnull = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pairs[i].val = pstrdup(sv2cstr(value));
|
|
|
|
pairs[i].vallen = hstoreCheckValLen(strlen(pairs[i].val));
|
|
|
|
pairs[i].isnull = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
pcount = hstoreUniquePairs(pairs, pcount, &buflen);
|
|
|
|
out = hstorePairs(pairs, pcount, buflen);
|
|
|
|
PG_RETURN_POINTER(out);
|
|
|
|
}
|