Commit c12d570f authored by Tom Lane's avatar Tom Lane

Support arrays over domains.

Allowing arrays with a domain type as their element type was left un-done
in the original domain patch, but not for any very good reason.  This
omission leads to such surprising results as array_agg() not working on
a domain column, because the parser can't identify a suitable output type
for the polymorphic aggregate.

In order to fix this, first clean up the APIs of coerce_to_domain() and
some internal functions in parse_coerce.c so that we consistently pass
around a CoercionContext along with CoercionForm.  Previously, we sometimes
passed an "isExplicit" boolean flag instead, which is strictly less
information; and coerce_to_domain() didn't even get that, but instead had
to reverse-engineer isExplicit from CoercionForm.  That's contrary to the
documentation in primnodes.h that says that CoercionForm only affects
display and not semantics.  I don't think this change fixes any live bugs,
but it makes things more consistent.  The main reason for doing it though
is that now build_coercion_expression() receives ccontext, which it needs
in order to be able to recursively invoke coerce_to_target_type().

Next, reimplement ArrayCoerceExpr so that the node does not directly know
any details of what has to be done to the individual array elements while
performing the array coercion.  Instead, the per-element processing is
represented by a sub-expression whose input is a source array element and
whose output is a target array element.  This simplifies life in
parse_coerce.c, because it can build that sub-expression by a recursive
invocation of coerce_to_target_type().  The executor now handles the
per-element processing as a compiled expression instead of hard-wired code.
The main advantage of this is that we can use a single ArrayCoerceExpr to
handle as many as three successive steps per element: base type conversion,
typmod coercion, and domain constraint checking.  The old code used two
stacked ArrayCoerceExprs to handle type + typmod coercion, which was pretty
inefficient, and adding yet another array deconstruction to do domain
constraint checking seemed very unappetizing.

In the case where we just need a single, very simple coercion function,
doing this straightforwardly leads to a noticeable increase in the
per-array-element runtime cost.  Hence, add an additional shortcut evalfunc
in execExprInterp.c that skips unnecessary overhead for that specific form
of expression.  The runtime speed of simple cases is within 1% or so of
where it was before, while cases that previously required two levels of
array processing are significantly faster.

Finally, create an implicit array type for every domain type, as we do for
base types, enums, etc.  Everything except the array-coercion case seems
to just work without further effort.

Tom Lane, reviewed by Andrew Dunstan

Discussion: https://postgr.es/m/9852.1499791473@sss.pgh.pa.us
parent 248e3375
...@@ -2630,6 +2630,7 @@ JumbleExpr(pgssJumbleState *jstate, Node *node) ...@@ -2630,6 +2630,7 @@ JumbleExpr(pgssJumbleState *jstate, Node *node)
APP_JUMB(acexpr->resulttype); APP_JUMB(acexpr->resulttype);
JumbleExpr(jstate, (Node *) acexpr->arg); JumbleExpr(jstate, (Node *) acexpr->arg);
JumbleExpr(jstate, (Node *) acexpr->elemexpr);
} }
break; break;
case T_ConvertRowtypeExpr: case T_ConvertRowtypeExpr:
......
...@@ -10,9 +10,8 @@ ...@@ -10,9 +10,8 @@
<para> <para>
<productname>PostgreSQL</productname> allows columns of a table to be <productname>PostgreSQL</productname> allows columns of a table to be
defined as variable-length multidimensional arrays. Arrays of any defined as variable-length multidimensional arrays. Arrays of any
built-in or user-defined base type, enum type, or composite type built-in or user-defined base type, enum type, composite type, range type,
can be created. or domain can be created.
Arrays of domains are not yet supported.
</para> </para>
<sect2 id="arrays-declaration"> <sect2 id="arrays-declaration">
......
...@@ -1738,11 +1738,14 @@ find_expr_references_walker(Node *node, ...@@ -1738,11 +1738,14 @@ find_expr_references_walker(Node *node,
{ {
ArrayCoerceExpr *acoerce = (ArrayCoerceExpr *) node; ArrayCoerceExpr *acoerce = (ArrayCoerceExpr *) node;
if (OidIsValid(acoerce->elemfuncid)) /* as above, depend on type */
add_object_address(OCLASS_PROC, acoerce->elemfuncid, 0,
context->addrs);
add_object_address(OCLASS_TYPE, acoerce->resulttype, 0, add_object_address(OCLASS_TYPE, acoerce->resulttype, 0,
context->addrs); context->addrs);
/* the collation might not be referenced anywhere else, either */
if (OidIsValid(acoerce->resultcollid) &&
acoerce->resultcollid != DEFAULT_COLLATION_OID)
add_object_address(OCLASS_COLLATION, acoerce->resultcollid, 0,
context->addrs);
/* fall through to examine arguments */ /* fall through to examine arguments */
} }
else if (IsA(node, ConvertRowtypeExpr)) else if (IsA(node, ConvertRowtypeExpr))
......
...@@ -729,6 +729,7 @@ ObjectAddress ...@@ -729,6 +729,7 @@ ObjectAddress
DefineDomain(CreateDomainStmt *stmt) DefineDomain(CreateDomainStmt *stmt)
{ {
char *domainName; char *domainName;
char *domainArrayName;
Oid domainNamespace; Oid domainNamespace;
AclResult aclresult; AclResult aclresult;
int16 internalLength; int16 internalLength;
...@@ -757,6 +758,7 @@ DefineDomain(CreateDomainStmt *stmt) ...@@ -757,6 +758,7 @@ DefineDomain(CreateDomainStmt *stmt)
Oid basetypeoid; Oid basetypeoid;
Oid old_type_oid; Oid old_type_oid;
Oid domaincoll; Oid domaincoll;
Oid domainArrayOid;
Form_pg_type baseType; Form_pg_type baseType;
int32 basetypeMod; int32 basetypeMod;
Oid baseColl; Oid baseColl;
...@@ -1027,6 +1029,9 @@ DefineDomain(CreateDomainStmt *stmt) ...@@ -1027,6 +1029,9 @@ DefineDomain(CreateDomainStmt *stmt)
} }
} }
/* Allocate OID for array type */
domainArrayOid = AssignTypeArrayOid();
/* /*
* Have TypeCreate do all the real work. * Have TypeCreate do all the real work.
*/ */
...@@ -1051,7 +1056,7 @@ DefineDomain(CreateDomainStmt *stmt) ...@@ -1051,7 +1056,7 @@ DefineDomain(CreateDomainStmt *stmt)
analyzeProcedure, /* analyze procedure */ analyzeProcedure, /* analyze procedure */
InvalidOid, /* no array element type */ InvalidOid, /* no array element type */
false, /* this isn't an array */ false, /* this isn't an array */
InvalidOid, /* no arrays for domains (yet) */ domainArrayOid, /* array type we are about to create */
basetypeoid, /* base type ID */ basetypeoid, /* base type ID */
defaultValue, /* default type value (text) */ defaultValue, /* default type value (text) */
defaultValueBin, /* default type value (binary) */ defaultValueBin, /* default type value (binary) */
...@@ -1063,6 +1068,48 @@ DefineDomain(CreateDomainStmt *stmt) ...@@ -1063,6 +1068,48 @@ DefineDomain(CreateDomainStmt *stmt)
typNotNull, /* Type NOT NULL */ typNotNull, /* Type NOT NULL */
domaincoll); /* type's collation */ domaincoll); /* type's collation */
/*
* Create the array type that goes with it.
*/
domainArrayName = makeArrayTypeName(domainName, domainNamespace);
/* alignment must be 'i' or 'd' for arrays */
alignment = (alignment == 'd') ? 'd' : 'i';
TypeCreate(domainArrayOid, /* force assignment of this type OID */
domainArrayName, /* type name */
domainNamespace, /* namespace */
InvalidOid, /* relation oid (n/a here) */
0, /* relation kind (ditto) */
GetUserId(), /* owner's ID */
-1, /* internal size (always varlena) */
TYPTYPE_BASE, /* type-type (base type) */
TYPCATEGORY_ARRAY, /* type-category (array) */
false, /* array types are never preferred */
delimiter, /* array element delimiter */
F_ARRAY_IN, /* input procedure */
F_ARRAY_OUT, /* output procedure */
F_ARRAY_RECV, /* receive procedure */
F_ARRAY_SEND, /* send procedure */
InvalidOid, /* typmodin procedure - none */
InvalidOid, /* typmodout procedure - none */
F_ARRAY_TYPANALYZE, /* analyze procedure */
address.objectId, /* element type ID */
true, /* yes this is an array type */
InvalidOid, /* no further array type */
InvalidOid, /* base type ID */
NULL, /* never a default type value */
NULL, /* binary default isn't sent either */
false, /* never passed by value */
alignment, /* see above */
'x', /* ARRAY is always toastable */
-1, /* typMod (Domains only) */
0, /* Array dimensions of typbasetype */
false, /* Type NOT NULL */
domaincoll); /* type's collation */
pfree(domainArrayName);
/* /*
* Process constraints which refer to the domain ID returned by TypeCreate * Process constraints which refer to the domain ID returned by TypeCreate
*/ */
...@@ -1139,6 +1186,7 @@ DefineEnum(CreateEnumStmt *stmt) ...@@ -1139,6 +1186,7 @@ DefineEnum(CreateEnumStmt *stmt)
errmsg("type \"%s\" already exists", enumName))); errmsg("type \"%s\" already exists", enumName)));
} }
/* Allocate OID for array type */
enumArrayOid = AssignTypeArrayOid(); enumArrayOid = AssignTypeArrayOid();
/* Create the pg_type entry */ /* Create the pg_type entry */
......
...@@ -1225,6 +1225,7 @@ ExecInitExprRec(Expr *node, PlanState *parent, ExprState *state, ...@@ -1225,6 +1225,7 @@ ExecInitExprRec(Expr *node, PlanState *parent, ExprState *state,
{ {
ArrayCoerceExpr *acoerce = (ArrayCoerceExpr *) node; ArrayCoerceExpr *acoerce = (ArrayCoerceExpr *) node;
Oid resultelemtype; Oid resultelemtype;
ExprState *elemstate;
/* evaluate argument into step's result area */ /* evaluate argument into step's result area */
ExecInitExprRec(acoerce->arg, parent, state, resv, resnull); ExecInitExprRec(acoerce->arg, parent, state, resv, resnull);
...@@ -1234,42 +1235,49 @@ ExecInitExprRec(Expr *node, PlanState *parent, ExprState *state, ...@@ -1234,42 +1235,49 @@ ExecInitExprRec(Expr *node, PlanState *parent, ExprState *state,
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("target type is not an array"))); errmsg("target type is not an array")));
/* Arrays over domains aren't supported yet */
Assert(getBaseType(resultelemtype) == resultelemtype);
scratch.opcode = EEOP_ARRAYCOERCE; /*
scratch.d.arraycoerce.coerceexpr = acoerce; * Construct a sub-expression for the per-element expression;
scratch.d.arraycoerce.resultelemtype = resultelemtype; * but don't ready it until after we check it for triviality.
* We assume it hasn't any Var references, but does have a
* CaseTestExpr representing the source array element values.
*/
elemstate = makeNode(ExprState);
elemstate->expr = acoerce->elemexpr;
elemstate->innermost_caseval = (Datum *) palloc(sizeof(Datum));
elemstate->innermost_casenull = (bool *) palloc(sizeof(bool));
if (OidIsValid(acoerce->elemfuncid)) ExecInitExprRec(acoerce->elemexpr, parent, elemstate,
{ &elemstate->resvalue, &elemstate->resnull);
AclResult aclresult;
/* Check permission to call function */ if (elemstate->steps_len == 1 &&
aclresult = pg_proc_aclcheck(acoerce->elemfuncid, elemstate->steps[0].opcode == EEOP_CASE_TESTVAL)
GetUserId(), {
ACL_EXECUTE); /* Trivial, so we need no per-element work at runtime */
if (aclresult != ACLCHECK_OK) elemstate = NULL;
aclcheck_error(aclresult, ACL_KIND_PROC, }
get_func_name(acoerce->elemfuncid)); else
InvokeFunctionExecuteHook(acoerce->elemfuncid); {
/* Not trivial, so append a DONE step */
scratch.opcode = EEOP_DONE;
ExprEvalPushStep(elemstate, &scratch);
/* and ready the subexpression */
ExecReadyExpr(elemstate);
}
/* Set up the primary fmgr lookup information */ scratch.opcode = EEOP_ARRAYCOERCE;
scratch.d.arraycoerce.elemfunc = scratch.d.arraycoerce.elemexprstate = elemstate;
(FmgrInfo *) palloc0(sizeof(FmgrInfo)); scratch.d.arraycoerce.resultelemtype = resultelemtype;
fmgr_info(acoerce->elemfuncid,
scratch.d.arraycoerce.elemfunc);
fmgr_info_set_expr((Node *) acoerce,
scratch.d.arraycoerce.elemfunc);
if (elemstate)
{
/* Set up workspace for array_map */ /* Set up workspace for array_map */
scratch.d.arraycoerce.amstate = scratch.d.arraycoerce.amstate =
(ArrayMapState *) palloc0(sizeof(ArrayMapState)); (ArrayMapState *) palloc0(sizeof(ArrayMapState));
} }
else else
{ {
/* Don't need workspace if there's no conversion func */ /* Don't need workspace if there's no subexpression */
scratch.d.arraycoerce.elemfunc = NULL;
scratch.d.arraycoerce.amstate = NULL; scratch.d.arraycoerce.amstate = NULL;
} }
......
...@@ -34,10 +34,8 @@ ...@@ -34,10 +34,8 @@
* *
* For very simple instructions the overhead of the full interpreter * For very simple instructions the overhead of the full interpreter
* "startup", as minimal as it is, is noticeable. Therefore * "startup", as minimal as it is, is noticeable. Therefore
* ExecReadyInterpretedExpr will choose to implement simple scalar Var * ExecReadyInterpretedExpr will choose to implement certain simple
* and Const expressions using special fast-path routines (ExecJust*). * opcode patterns using special fast-path routines (ExecJust*).
* Benchmarking shows anything more complex than those may as well use the
* "full interpreter".
* *
* Complex or uncommon instructions are not implemented in-line in * Complex or uncommon instructions are not implemented in-line in
* ExecInterpExpr(), rather we call out to a helper function appearing later * ExecInterpExpr(), rather we call out to a helper function appearing later
...@@ -149,6 +147,7 @@ static Datum ExecJustConst(ExprState *state, ExprContext *econtext, bool *isnull ...@@ -149,6 +147,7 @@ static Datum ExecJustConst(ExprState *state, ExprContext *econtext, bool *isnull
static Datum ExecJustAssignInnerVar(ExprState *state, ExprContext *econtext, bool *isnull); static Datum ExecJustAssignInnerVar(ExprState *state, ExprContext *econtext, bool *isnull);
static Datum ExecJustAssignOuterVar(ExprState *state, ExprContext *econtext, bool *isnull); static Datum ExecJustAssignOuterVar(ExprState *state, ExprContext *econtext, bool *isnull);
static Datum ExecJustAssignScanVar(ExprState *state, ExprContext *econtext, bool *isnull); static Datum ExecJustAssignScanVar(ExprState *state, ExprContext *econtext, bool *isnull);
static Datum ExecJustApplyFuncToCase(ExprState *state, ExprContext *econtext, bool *isnull);
/* /*
...@@ -184,10 +183,8 @@ ExecReadyInterpretedExpr(ExprState *state) ...@@ -184,10 +183,8 @@ ExecReadyInterpretedExpr(ExprState *state)
/* /*
* Select fast-path evalfuncs for very simple expressions. "Starting up" * Select fast-path evalfuncs for very simple expressions. "Starting up"
* the full interpreter is a measurable overhead for these. Plain Vars * the full interpreter is a measurable overhead for these, and these
* and Const seem to be the only ones where the intrinsic cost is small * patterns occur often enough to be worth optimizing.
* enough that the overhead of ExecInterpExpr matters. For more complex
* expressions it's cheaper to use ExecInterpExpr always.
*/ */
if (state->steps_len == 3) if (state->steps_len == 3)
{ {
...@@ -230,6 +227,13 @@ ExecReadyInterpretedExpr(ExprState *state) ...@@ -230,6 +227,13 @@ ExecReadyInterpretedExpr(ExprState *state)
state->evalfunc = ExecJustAssignScanVar; state->evalfunc = ExecJustAssignScanVar;
return; return;
} }
else if (step0 == EEOP_CASE_TESTVAL &&
step1 == EEOP_FUNCEXPR_STRICT &&
state->steps[0].d.casetest.value)
{
state->evalfunc = ExecJustApplyFuncToCase;
return;
}
} }
else if (state->steps_len == 2 && else if (state->steps_len == 2 &&
state->steps[0].opcode == EEOP_CONST) state->steps[0].opcode == EEOP_CONST)
...@@ -1275,7 +1279,7 @@ ExecInterpExpr(ExprState *state, ExprContext *econtext, bool *isnull) ...@@ -1275,7 +1279,7 @@ ExecInterpExpr(ExprState *state, ExprContext *econtext, bool *isnull)
EEO_CASE(EEOP_ARRAYCOERCE) EEO_CASE(EEOP_ARRAYCOERCE)
{ {
/* too complex for an inline implementation */ /* too complex for an inline implementation */
ExecEvalArrayCoerce(state, op); ExecEvalArrayCoerce(state, op, econtext);
EEO_NEXT(); EEO_NEXT();
} }
...@@ -1811,6 +1815,43 @@ ExecJustAssignScanVar(ExprState *state, ExprContext *econtext, bool *isnull) ...@@ -1811,6 +1815,43 @@ ExecJustAssignScanVar(ExprState *state, ExprContext *econtext, bool *isnull)
return 0; return 0;
} }
/* Evaluate CASE_TESTVAL and apply a strict function to it */
static Datum
ExecJustApplyFuncToCase(ExprState *state, ExprContext *econtext, bool *isnull)
{
ExprEvalStep *op = &state->steps[0];
FunctionCallInfo fcinfo;
bool *argnull;
int argno;
Datum d;
/*
* XXX with some redesign of the CaseTestExpr mechanism, maybe we could
* get rid of this data shuffling?
*/
*op->resvalue = *op->d.casetest.value;
*op->resnull = *op->d.casetest.isnull;
op++;
fcinfo = op->d.func.fcinfo_data;
argnull = fcinfo->argnull;
/* strict function, so check for NULL args */
for (argno = 0; argno < op->d.func.nargs; argno++)
{
if (argnull[argno])
{
*isnull = true;
return (Datum) 0;
}
}
fcinfo->isnull = false;
d = op->d.func.fn_addr(fcinfo);
*isnull = fcinfo->isnull;
return d;
}
/* /*
* Do one-time initialization of interpretation machinery. * Do one-time initialization of interpretation machinery.
...@@ -2345,11 +2386,9 @@ ExecEvalArrayExpr(ExprState *state, ExprEvalStep *op) ...@@ -2345,11 +2386,9 @@ ExecEvalArrayExpr(ExprState *state, ExprEvalStep *op)
* Source array is in step's result variable. * Source array is in step's result variable.
*/ */
void void
ExecEvalArrayCoerce(ExprState *state, ExprEvalStep *op) ExecEvalArrayCoerce(ExprState *state, ExprEvalStep *op, ExprContext *econtext)
{ {
ArrayCoerceExpr *acoerce = op->d.arraycoerce.coerceexpr;
Datum arraydatum; Datum arraydatum;
FunctionCallInfoData locfcinfo;
/* NULL array -> NULL result */ /* NULL array -> NULL result */
if (*op->resnull) if (*op->resnull)
...@@ -2361,7 +2400,7 @@ ExecEvalArrayCoerce(ExprState *state, ExprEvalStep *op) ...@@ -2361,7 +2400,7 @@ ExecEvalArrayCoerce(ExprState *state, ExprEvalStep *op)
* If it's binary-compatible, modify the element type in the array header, * If it's binary-compatible, modify the element type in the array header,
* but otherwise leave the array as we received it. * but otherwise leave the array as we received it.
*/ */
if (!OidIsValid(acoerce->elemfuncid)) if (op->d.arraycoerce.elemexprstate == NULL)
{ {
/* Detoast input array if necessary, and copy in any case */ /* Detoast input array if necessary, and copy in any case */
ArrayType *array = DatumGetArrayTypePCopy(arraydatum); ArrayType *array = DatumGetArrayTypePCopy(arraydatum);
...@@ -2372,23 +2411,12 @@ ExecEvalArrayCoerce(ExprState *state, ExprEvalStep *op) ...@@ -2372,23 +2411,12 @@ ExecEvalArrayCoerce(ExprState *state, ExprEvalStep *op)
} }
/* /*
* Use array_map to apply the function to each array element. * Use array_map to apply the sub-expression to each array element.
*
* We pass on the desttypmod and isExplicit flags whether or not the
* function wants them.
*
* Note: coercion functions are assumed to not use collation.
*/ */
InitFunctionCallInfoData(locfcinfo, op->d.arraycoerce.elemfunc, 3, *op->resvalue = array_map(arraydatum,
InvalidOid, NULL, NULL); op->d.arraycoerce.elemexprstate,
locfcinfo.arg[0] = arraydatum; econtext,
locfcinfo.arg[1] = Int32GetDatum(acoerce->resulttypmod); op->d.arraycoerce.resultelemtype,
locfcinfo.arg[2] = BoolGetDatum(acoerce->isExplicit);
locfcinfo.argnull[0] = false;
locfcinfo.argnull[1] = false;
locfcinfo.argnull[2] = false;
*op->resvalue = array_map(&locfcinfo, op->d.arraycoerce.resultelemtype,
op->d.arraycoerce.amstate); op->d.arraycoerce.amstate);
} }
......
...@@ -1698,11 +1698,10 @@ _copyArrayCoerceExpr(const ArrayCoerceExpr *from) ...@@ -1698,11 +1698,10 @@ _copyArrayCoerceExpr(const ArrayCoerceExpr *from)
ArrayCoerceExpr *newnode = makeNode(ArrayCoerceExpr); ArrayCoerceExpr *newnode = makeNode(ArrayCoerceExpr);
COPY_NODE_FIELD(arg); COPY_NODE_FIELD(arg);
COPY_SCALAR_FIELD(elemfuncid); COPY_NODE_FIELD(elemexpr);
COPY_SCALAR_FIELD(resulttype); COPY_SCALAR_FIELD(resulttype);
COPY_SCALAR_FIELD(resulttypmod); COPY_SCALAR_FIELD(resulttypmod);
COPY_SCALAR_FIELD(resultcollid); COPY_SCALAR_FIELD(resultcollid);
COPY_SCALAR_FIELD(isExplicit);
COPY_SCALAR_FIELD(coerceformat); COPY_SCALAR_FIELD(coerceformat);
COPY_LOCATION_FIELD(location); COPY_LOCATION_FIELD(location);
......
...@@ -513,11 +513,10 @@ static bool ...@@ -513,11 +513,10 @@ static bool
_equalArrayCoerceExpr(const ArrayCoerceExpr *a, const ArrayCoerceExpr *b) _equalArrayCoerceExpr(const ArrayCoerceExpr *a, const ArrayCoerceExpr *b)
{ {
COMPARE_NODE_FIELD(arg); COMPARE_NODE_FIELD(arg);
COMPARE_SCALAR_FIELD(elemfuncid); COMPARE_NODE_FIELD(elemexpr);
COMPARE_SCALAR_FIELD(resulttype); COMPARE_SCALAR_FIELD(resulttype);
COMPARE_SCALAR_FIELD(resulttypmod); COMPARE_SCALAR_FIELD(resulttypmod);
COMPARE_SCALAR_FIELD(resultcollid); COMPARE_SCALAR_FIELD(resultcollid);
COMPARE_SCALAR_FIELD(isExplicit);
COMPARE_COERCIONFORM_FIELD(coerceformat); COMPARE_COERCIONFORM_FIELD(coerceformat);
COMPARE_LOCATION_FIELD(location); COMPARE_LOCATION_FIELD(location);
......
...@@ -1717,15 +1717,6 @@ check_functions_in_node(Node *node, check_function_callback checker, ...@@ -1717,15 +1717,6 @@ check_functions_in_node(Node *node, check_function_callback checker,
return true; return true;
} }
break; break;
case T_ArrayCoerceExpr:
{
ArrayCoerceExpr *expr = (ArrayCoerceExpr *) node;
if (OidIsValid(expr->elemfuncid) &&
checker(expr->elemfuncid, context))
return true;
}
break;
case T_RowCompareExpr: case T_RowCompareExpr:
{ {
RowCompareExpr *rcexpr = (RowCompareExpr *) node; RowCompareExpr *rcexpr = (RowCompareExpr *) node;
...@@ -2023,7 +2014,15 @@ expression_tree_walker(Node *node, ...@@ -2023,7 +2014,15 @@ expression_tree_walker(Node *node,
case T_CoerceViaIO: case T_CoerceViaIO:
return walker(((CoerceViaIO *) node)->arg, context); return walker(((CoerceViaIO *) node)->arg, context);
case T_ArrayCoerceExpr: case T_ArrayCoerceExpr:
return walker(((ArrayCoerceExpr *) node)->arg, context); {
ArrayCoerceExpr *acoerce = (ArrayCoerceExpr *) node;
if (walker(acoerce->arg, context))
return true;
if (walker(acoerce->elemexpr, context))
return true;
}
break;
case T_ConvertRowtypeExpr: case T_ConvertRowtypeExpr:
return walker(((ConvertRowtypeExpr *) node)->arg, context); return walker(((ConvertRowtypeExpr *) node)->arg, context);
case T_CollateExpr: case T_CollateExpr:
...@@ -2705,6 +2704,7 @@ expression_tree_mutator(Node *node, ...@@ -2705,6 +2704,7 @@ expression_tree_mutator(Node *node,
FLATCOPY(newnode, acoerce, ArrayCoerceExpr); FLATCOPY(newnode, acoerce, ArrayCoerceExpr);
MUTATE(newnode->arg, acoerce->arg, Expr *); MUTATE(newnode->arg, acoerce->arg, Expr *);
MUTATE(newnode->elemexpr, acoerce->elemexpr, Expr *);
return (Node *) newnode; return (Node *) newnode;
} }
break; break;
......
...@@ -1394,11 +1394,10 @@ _outArrayCoerceExpr(StringInfo str, const ArrayCoerceExpr *node) ...@@ -1394,11 +1394,10 @@ _outArrayCoerceExpr(StringInfo str, const ArrayCoerceExpr *node)
WRITE_NODE_TYPE("ARRAYCOERCEEXPR"); WRITE_NODE_TYPE("ARRAYCOERCEEXPR");
WRITE_NODE_FIELD(arg); WRITE_NODE_FIELD(arg);
WRITE_OID_FIELD(elemfuncid); WRITE_NODE_FIELD(elemexpr);
WRITE_OID_FIELD(resulttype); WRITE_OID_FIELD(resulttype);
WRITE_INT_FIELD(resulttypmod); WRITE_INT_FIELD(resulttypmod);
WRITE_OID_FIELD(resultcollid); WRITE_OID_FIELD(resultcollid);
WRITE_BOOL_FIELD(isExplicit);
WRITE_ENUM_FIELD(coerceformat, CoercionForm); WRITE_ENUM_FIELD(coerceformat, CoercionForm);
WRITE_LOCATION_FIELD(location); WRITE_LOCATION_FIELD(location);
} }
......
...@@ -892,11 +892,10 @@ _readArrayCoerceExpr(void) ...@@ -892,11 +892,10 @@ _readArrayCoerceExpr(void)
READ_LOCALS(ArrayCoerceExpr); READ_LOCALS(ArrayCoerceExpr);
READ_NODE_FIELD(arg); READ_NODE_FIELD(arg);
READ_OID_FIELD(elemfuncid); READ_NODE_FIELD(elemexpr);
READ_OID_FIELD(resulttype); READ_OID_FIELD(resulttype);
READ_INT_FIELD(resulttypmod); READ_INT_FIELD(resulttypmod);
READ_OID_FIELD(resultcollid); READ_OID_FIELD(resultcollid);
READ_BOOL_FIELD(isExplicit);
READ_ENUM_FIELD(coerceformat, CoercionForm); READ_ENUM_FIELD(coerceformat, CoercionForm);
READ_LOCATION_FIELD(location); READ_LOCATION_FIELD(location);
......
...@@ -3632,11 +3632,14 @@ cost_qual_eval_walker(Node *node, cost_qual_eval_context *context) ...@@ -3632,11 +3632,14 @@ cost_qual_eval_walker(Node *node, cost_qual_eval_context *context)
else if (IsA(node, ArrayCoerceExpr)) else if (IsA(node, ArrayCoerceExpr))
{ {
ArrayCoerceExpr *acoerce = (ArrayCoerceExpr *) node; ArrayCoerceExpr *acoerce = (ArrayCoerceExpr *) node;
Node *arraynode = (Node *) acoerce->arg; QualCost perelemcost;
if (OidIsValid(acoerce->elemfuncid)) cost_qual_eval_node(&perelemcost, (Node *) acoerce->elemexpr,
context->total.per_tuple += get_func_cost(acoerce->elemfuncid) * context->root);
cpu_operator_cost * estimate_array_length(arraynode); context->total.startup += perelemcost.startup;
if (perelemcost.per_tuple > 0)
context->total.per_tuple += perelemcost.per_tuple *
estimate_array_length((Node *) acoerce->arg);
} }
else if (IsA(node, RowCompareExpr)) else if (IsA(node, RowCompareExpr))
{ {
......
...@@ -1395,12 +1395,6 @@ fix_expr_common(PlannerInfo *root, Node *node) ...@@ -1395,12 +1395,6 @@ fix_expr_common(PlannerInfo *root, Node *node)
record_plan_function_dependency(root, record_plan_function_dependency(root,
((ScalarArrayOpExpr *) node)->opfuncid); ((ScalarArrayOpExpr *) node)->opfuncid);
} }
else if (IsA(node, ArrayCoerceExpr))
{
if (OidIsValid(((ArrayCoerceExpr *) node)->elemfuncid))
record_plan_function_dependency(root,
((ArrayCoerceExpr *) node)->elemfuncid);
}
else if (IsA(node, Const)) else if (IsA(node, Const))
{ {
Const *con = (Const *) node; Const *con = (Const *) node;
......
...@@ -306,9 +306,9 @@ expand_targetlist(List *tlist, int command_type, ...@@ -306,9 +306,9 @@ expand_targetlist(List *tlist, int command_type,
new_expr = coerce_to_domain(new_expr, new_expr = coerce_to_domain(new_expr,
InvalidOid, -1, InvalidOid, -1,
atttype, atttype,
COERCION_IMPLICIT,
COERCE_IMPLICIT_CAST, COERCE_IMPLICIT_CAST,
-1, -1,
false,
false); false);
} }
else else
......
...@@ -1361,6 +1361,17 @@ contain_nonstrict_functions_walker(Node *node, void *context) ...@@ -1361,6 +1361,17 @@ contain_nonstrict_functions_walker(Node *node, void *context)
return true; return true;
if (IsA(node, FieldStore)) if (IsA(node, FieldStore))
return true; return true;
if (IsA(node, ArrayCoerceExpr))
{
/*
* ArrayCoerceExpr is strict at the array level, regardless of what
* the per-element expression is; so we should ignore elemexpr and
* recurse only into the arg.
*/
return expression_tree_walker((Node *) ((ArrayCoerceExpr *) node)->arg,
contain_nonstrict_functions_walker,
context);
}
if (IsA(node, CaseExpr)) if (IsA(node, CaseExpr))
return true; return true;
if (IsA(node, ArrayExpr)) if (IsA(node, ArrayExpr))
...@@ -1380,14 +1391,11 @@ contain_nonstrict_functions_walker(Node *node, void *context) ...@@ -1380,14 +1391,11 @@ contain_nonstrict_functions_walker(Node *node, void *context)
if (IsA(node, BooleanTest)) if (IsA(node, BooleanTest))
return true; return true;
/* /* Check other function-containing nodes */
* Check other function-containing nodes; but ArrayCoerceExpr is strict at if (check_functions_in_node(node, contain_nonstrict_functions_checker,
* the array level, regardless of elemfunc.
*/
if (!IsA(node, ArrayCoerceExpr) &&
check_functions_in_node(node, contain_nonstrict_functions_checker,
context)) context))
return true; return true;
return expression_tree_walker(node, contain_nonstrict_functions_walker, return expression_tree_walker(node, contain_nonstrict_functions_walker,
context); context);
} }
...@@ -1757,7 +1765,7 @@ find_nonnullable_rels_walker(Node *node, bool top_level) ...@@ -1757,7 +1765,7 @@ find_nonnullable_rels_walker(Node *node, bool top_level)
} }
else if (IsA(node, ArrayCoerceExpr)) else if (IsA(node, ArrayCoerceExpr))
{ {
/* ArrayCoerceExpr is strict at the array level */ /* ArrayCoerceExpr is strict at the array level; ignore elemexpr */
ArrayCoerceExpr *expr = (ArrayCoerceExpr *) node; ArrayCoerceExpr *expr = (ArrayCoerceExpr *) node;
result = find_nonnullable_rels_walker((Node *) expr->arg, top_level); result = find_nonnullable_rels_walker((Node *) expr->arg, top_level);
...@@ -1965,7 +1973,7 @@ find_nonnullable_vars_walker(Node *node, bool top_level) ...@@ -1965,7 +1973,7 @@ find_nonnullable_vars_walker(Node *node, bool top_level)
} }
else if (IsA(node, ArrayCoerceExpr)) else if (IsA(node, ArrayCoerceExpr))
{ {
/* ArrayCoerceExpr is strict at the array level */ /* ArrayCoerceExpr is strict at the array level; ignore elemexpr */
ArrayCoerceExpr *expr = (ArrayCoerceExpr *) node; ArrayCoerceExpr *expr = (ArrayCoerceExpr *) node;
result = find_nonnullable_vars_walker((Node *) expr->arg, top_level); result = find_nonnullable_vars_walker((Node *) expr->arg, top_level);
...@@ -3005,32 +3013,38 @@ eval_const_expressions_mutator(Node *node, ...@@ -3005,32 +3013,38 @@ eval_const_expressions_mutator(Node *node,
{ {
ArrayCoerceExpr *expr = (ArrayCoerceExpr *) node; ArrayCoerceExpr *expr = (ArrayCoerceExpr *) node;
Expr *arg; Expr *arg;
Expr *elemexpr;
ArrayCoerceExpr *newexpr; ArrayCoerceExpr *newexpr;
/* /*
* Reduce constants in the ArrayCoerceExpr's argument, then * Reduce constants in the ArrayCoerceExpr's argument and
* build a new ArrayCoerceExpr. * per-element expressions, then build a new ArrayCoerceExpr.
*/ */
arg = (Expr *) eval_const_expressions_mutator((Node *) expr->arg, arg = (Expr *) eval_const_expressions_mutator((Node *) expr->arg,
context); context);
elemexpr = (Expr *) eval_const_expressions_mutator((Node *) expr->elemexpr,
context);
newexpr = makeNode(ArrayCoerceExpr); newexpr = makeNode(ArrayCoerceExpr);
newexpr->arg = arg; newexpr->arg = arg;
newexpr->elemfuncid = expr->elemfuncid; newexpr->elemexpr = elemexpr;
newexpr->resulttype = expr->resulttype; newexpr->resulttype = expr->resulttype;
newexpr->resulttypmod = expr->resulttypmod; newexpr->resulttypmod = expr->resulttypmod;
newexpr->resultcollid = expr->resultcollid; newexpr->resultcollid = expr->resultcollid;
newexpr->isExplicit = expr->isExplicit;
newexpr->coerceformat = expr->coerceformat; newexpr->coerceformat = expr->coerceformat;
newexpr->location = expr->location; newexpr->location = expr->location;
/* /*
* If constant argument and it's a binary-coercible or * If constant argument and per-element expression is
* immutable conversion, we can simplify it to a constant. * immutable, we can simplify the whole thing to a constant.
* Exception: although contain_mutable_functions considers
* CoerceToDomain immutable for historical reasons, let's not
* do so here; this ensures coercion to an array-over-domain
* does not apply the domain's constraints until runtime.
*/ */
if (arg && IsA(arg, Const) && if (arg && IsA(arg, Const) &&
(!OidIsValid(newexpr->elemfuncid) || elemexpr && !IsA(elemexpr, CoerceToDomain) &&
func_volatile(newexpr->elemfuncid) == PROVOLATILE_IMMUTABLE)) !contain_mutable_functions((Node *) elemexpr))
return (Node *) evaluate_expr((Expr *) newexpr, return (Node *) evaluate_expr((Expr *) newexpr,
newexpr->resulttype, newexpr->resulttype,
newexpr->resulttypmod, newexpr->resulttypmod,
......
This diff is collapsed.
...@@ -875,9 +875,9 @@ rewriteTargetListIU(List *targetList, ...@@ -875,9 +875,9 @@ rewriteTargetListIU(List *targetList,
new_expr = coerce_to_domain(new_expr, new_expr = coerce_to_domain(new_expr,
InvalidOid, -1, InvalidOid, -1,
att_tup->atttypid, att_tup->atttypid,
COERCION_IMPLICIT,
COERCE_IMPLICIT_CAST, COERCE_IMPLICIT_CAST,
-1, -1,
false,
false); false);
} }
} }
...@@ -1271,9 +1271,9 @@ rewriteValuesRTE(RangeTblEntry *rte, Relation target_relation, List *attrnos) ...@@ -1271,9 +1271,9 @@ rewriteValuesRTE(RangeTblEntry *rte, Relation target_relation, List *attrnos)
new_expr = coerce_to_domain(new_expr, new_expr = coerce_to_domain(new_expr,
InvalidOid, -1, InvalidOid, -1,
att_tup->atttypid, att_tup->atttypid,
COERCION_IMPLICIT,
COERCE_IMPLICIT_CAST, COERCE_IMPLICIT_CAST,
-1, -1,
false,
false); false);
} }
newList = lappend(newList, new_expr); newList = lappend(newList, new_expr);
......
...@@ -1429,9 +1429,9 @@ ReplaceVarsFromTargetList_callback(Var *var, ...@@ -1429,9 +1429,9 @@ ReplaceVarsFromTargetList_callback(Var *var,
var->varcollid), var->varcollid),
InvalidOid, -1, InvalidOid, -1,
var->vartype, var->vartype,
COERCION_IMPLICIT,
COERCE_IMPLICIT_CAST, COERCE_IMPLICIT_CAST,
-1, -1,
false,
false); false);
} }
elog(ERROR, "could not find replacement targetlist entry for attno %d", elog(ERROR, "could not find replacement targetlist entry for attno %d",
......
...@@ -3092,21 +3092,18 @@ array_set(ArrayType *array, int nSubscripts, int *indx, ...@@ -3092,21 +3092,18 @@ array_set(ArrayType *array, int nSubscripts, int *indx,
/* /*
* array_map() * array_map()
* *
* Map an array through an arbitrary function. Return a new array with * Map an array through an arbitrary expression. Return a new array with
* same dimensions and each source element transformed by fn(). Each * the same dimensions and each source element transformed by the given,
* source element is passed as the first argument to fn(); additional * already-compiled expression. Each source element is placed in the
* arguments to be passed to fn() can be specified by the caller. * innermost_caseval/innermost_casenull fields of the ExprState.
* The output array can have a different element type than the input.
* *
* Parameters are: * Parameters are:
* * fcinfo: a function-call data structure pre-constructed by the caller * * arrayd: Datum representing array argument.
* to be ready to call the desired function, with everything except the * * exprstate: ExprState representing the per-element transformation.
* first argument position filled in. In particular, flinfo identifies * * econtext: context for expression evaluation.
* the function fn(), and if nargs > 1 then argument positions after the
* first must be preset to the additional values to be passed. The
* first argument position initially holds the input array value.
* * retType: OID of element type of output array. This must be the same as, * * retType: OID of element type of output array. This must be the same as,
* or binary-compatible with, the result type of fn(). * or binary-compatible with, the result type of the expression. It might
* be different from the input array's element type.
* * amstate: workspace for array_map. Must be zeroed by caller before * * amstate: workspace for array_map. Must be zeroed by caller before
* first call, and not touched after that. * first call, and not touched after that.
* *
...@@ -3116,11 +3113,14 @@ array_set(ArrayType *array, int nSubscripts, int *indx, ...@@ -3116,11 +3113,14 @@ array_set(ArrayType *array, int nSubscripts, int *indx,
* *
* NB: caller must assure that input array is not NULL. NULL elements in * NB: caller must assure that input array is not NULL. NULL elements in
* the array are OK however. * the array are OK however.
* NB: caller should be running in econtext's per-tuple memory context.
*/ */
Datum Datum
array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate) array_map(Datum arrayd,
ExprState *exprstate, ExprContext *econtext,
Oid retType, ArrayMapState *amstate)
{ {
AnyArrayType *v; AnyArrayType *v = DatumGetAnyArrayP(arrayd);
ArrayType *result; ArrayType *result;
Datum *values; Datum *values;
bool *nulls; bool *nulls;
...@@ -3141,13 +3141,8 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate) ...@@ -3141,13 +3141,8 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate)
array_iter iter; array_iter iter;
ArrayMetaState *inp_extra; ArrayMetaState *inp_extra;
ArrayMetaState *ret_extra; ArrayMetaState *ret_extra;
Datum *transform_source = exprstate->innermost_caseval;
/* Get input array */ bool *transform_source_isnull = exprstate->innermost_casenull;
if (fcinfo->nargs < 1)
elog(ERROR, "invalid nargs: %d", fcinfo->nargs);
if (PG_ARGISNULL(0))
elog(ERROR, "null input array");
v = PG_GETARG_ANY_ARRAY_P(0);
inpType = AARR_ELEMTYPE(v); inpType = AARR_ELEMTYPE(v);
ndim = AARR_NDIM(v); ndim = AARR_NDIM(v);
...@@ -3158,7 +3153,7 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate) ...@@ -3158,7 +3153,7 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate)
if (nitems <= 0) if (nitems <= 0)
{ {
/* Return empty array */ /* Return empty array */
PG_RETURN_ARRAYTYPE_P(construct_empty_array(retType)); return PointerGetDatum(construct_empty_array(retType));
} }
/* /*
...@@ -3203,39 +3198,15 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate) ...@@ -3203,39 +3198,15 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate)
for (i = 0; i < nitems; i++) for (i = 0; i < nitems; i++)
{ {
bool callit = true;
/* Get source element, checking for NULL */ /* Get source element, checking for NULL */
fcinfo->arg[0] = array_iter_next(&iter, &fcinfo->argnull[0], i, *transform_source =
inp_typlen, inp_typbyval, inp_typalign); array_iter_next(&iter, transform_source_isnull, i,
inp_typlen, inp_typbyval, inp_typalign);
/*
* Apply the given function to source elt and extra args.
*/
if (fcinfo->flinfo->fn_strict)
{
int j;
for (j = 0; j < fcinfo->nargs; j++) /* Apply the given expression to source element */
{ values[i] = ExecEvalExpr(exprstate, econtext, &nulls[i]);
if (fcinfo->argnull[j])
{
callit = false;
break;
}
}
}
if (callit) if (nulls[i])
{
fcinfo->isnull = false;
values[i] = FunctionCallInvoke(fcinfo);
}
else
fcinfo->isnull = true;
nulls[i] = fcinfo->isnull;
if (fcinfo->isnull)
hasnulls = true; hasnulls = true;
else else
{ {
...@@ -3254,7 +3225,7 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate) ...@@ -3254,7 +3225,7 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate)
} }
} }
/* Allocate and initialize the result array */ /* Allocate and fill the result array */
if (hasnulls) if (hasnulls)
{ {
dataoffset = ARR_OVERHEAD_WITHNULLS(ndim, nitems); dataoffset = ARR_OVERHEAD_WITHNULLS(ndim, nitems);
...@@ -3273,18 +3244,18 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate) ...@@ -3273,18 +3244,18 @@ array_map(FunctionCallInfo fcinfo, Oid retType, ArrayMapState *amstate)
memcpy(ARR_DIMS(result), AARR_DIMS(v), ndim * sizeof(int)); memcpy(ARR_DIMS(result), AARR_DIMS(v), ndim * sizeof(int));
memcpy(ARR_LBOUND(result), AARR_LBOUND(v), ndim * sizeof(int)); memcpy(ARR_LBOUND(result), AARR_LBOUND(v), ndim * sizeof(int));
/*
* Note: do not risk trying to pfree the results of the called function
*/
CopyArrayEls(result, CopyArrayEls(result,
values, nulls, nitems, values, nulls, nitems,
typlen, typbyval, typalign, typlen, typbyval, typalign,
false); false);
/*
* Note: do not risk trying to pfree the results of the called expression
*/
pfree(values); pfree(values);
pfree(nulls); pfree(nulls);
PG_RETURN_ARRAYTYPE_P(result); return PointerGetDatum(result);
} }
/* /*
......
...@@ -1816,10 +1816,19 @@ strip_array_coercion(Node *node) ...@@ -1816,10 +1816,19 @@ strip_array_coercion(Node *node)
{ {
for (;;) for (;;)
{ {
if (node && IsA(node, ArrayCoerceExpr) && if (node && IsA(node, ArrayCoerceExpr))
((ArrayCoerceExpr *) node)->elemfuncid == InvalidOid)
{ {
node = (Node *) ((ArrayCoerceExpr *) node)->arg; ArrayCoerceExpr *acoerce = (ArrayCoerceExpr *) node;
/*
* If the per-element expression is just a RelabelType on top of
* CaseTestExpr, then we know it's a binary-compatible relabeling.
*/
if (IsA(acoerce->elemexpr, RelabelType) &&
IsA(((RelabelType *) acoerce->elemexpr)->arg, CaseTestExpr))
node = (Node *) acoerce->arg;
else
break;
} }
else if (node && IsA(node, RelabelType)) else if (node && IsA(node, RelabelType))
{ {
......
...@@ -1941,8 +1941,6 @@ get_call_expr_argtype(Node *expr, int argnum) ...@@ -1941,8 +1941,6 @@ get_call_expr_argtype(Node *expr, int argnum)
args = ((DistinctExpr *) expr)->args; args = ((DistinctExpr *) expr)->args;
else if (IsA(expr, ScalarArrayOpExpr)) else if (IsA(expr, ScalarArrayOpExpr))
args = ((ScalarArrayOpExpr *) expr)->args; args = ((ScalarArrayOpExpr *) expr)->args;
else if (IsA(expr, ArrayCoerceExpr))
args = list_make1(((ArrayCoerceExpr *) expr)->arg);
else if (IsA(expr, NullIfExpr)) else if (IsA(expr, NullIfExpr))
args = ((NullIfExpr *) expr)->args; args = ((NullIfExpr *) expr)->args;
else if (IsA(expr, WindowFunc)) else if (IsA(expr, WindowFunc))
...@@ -1956,16 +1954,12 @@ get_call_expr_argtype(Node *expr, int argnum) ...@@ -1956,16 +1954,12 @@ get_call_expr_argtype(Node *expr, int argnum)
argtype = exprType((Node *) list_nth(args, argnum)); argtype = exprType((Node *) list_nth(args, argnum));
/* /*
* special hack for ScalarArrayOpExpr and ArrayCoerceExpr: what the * special hack for ScalarArrayOpExpr: what the underlying function will
* underlying function will actually get passed is the element type of the * actually get passed is the element type of the array.
* array.
*/ */
if (IsA(expr, ScalarArrayOpExpr) && if (IsA(expr, ScalarArrayOpExpr) &&
argnum == 1) argnum == 1)
argtype = get_base_element_type(argtype); argtype = get_base_element_type(argtype);
else if (IsA(expr, ArrayCoerceExpr) &&
argnum == 0)
argtype = get_base_element_type(argtype);
return argtype; return argtype;
} }
...@@ -2012,8 +2006,6 @@ get_call_expr_arg_stable(Node *expr, int argnum) ...@@ -2012,8 +2006,6 @@ get_call_expr_arg_stable(Node *expr, int argnum)
args = ((DistinctExpr *) expr)->args; args = ((DistinctExpr *) expr)->args;
else if (IsA(expr, ScalarArrayOpExpr)) else if (IsA(expr, ScalarArrayOpExpr))
args = ((ScalarArrayOpExpr *) expr)->args; args = ((ScalarArrayOpExpr *) expr)->args;
else if (IsA(expr, ArrayCoerceExpr))
args = list_make1(((ArrayCoerceExpr *) expr)->arg);
else if (IsA(expr, NullIfExpr)) else if (IsA(expr, NullIfExpr))
args = ((NullIfExpr *) expr)->args; args = ((NullIfExpr *) expr)->args;
else if (IsA(expr, WindowFunc)) else if (IsA(expr, WindowFunc))
......
...@@ -53,6 +53,6 @@ ...@@ -53,6 +53,6 @@
*/ */
/* yyyymmddN */ /* yyyymmddN */
#define CATALOG_VERSION_NO 201709191 #define CATALOG_VERSION_NO 201709301
#endif #endif
...@@ -385,10 +385,8 @@ typedef struct ExprEvalStep ...@@ -385,10 +385,8 @@ typedef struct ExprEvalStep
/* for EEOP_ARRAYCOERCE */ /* for EEOP_ARRAYCOERCE */
struct struct
{ {
ArrayCoerceExpr *coerceexpr; ExprState *elemexprstate; /* null if no per-element work */
Oid resultelemtype; /* element type of result array */ Oid resultelemtype; /* element type of result array */
FmgrInfo *elemfunc; /* lookup info for element coercion
* function */
struct ArrayMapState *amstate; /* workspace for array_map */ struct ArrayMapState *amstate; /* workspace for array_map */
} arraycoerce; } arraycoerce;
...@@ -621,7 +619,8 @@ extern void ExecEvalRowNull(ExprState *state, ExprEvalStep *op, ...@@ -621,7 +619,8 @@ extern void ExecEvalRowNull(ExprState *state, ExprEvalStep *op,
extern void ExecEvalRowNotNull(ExprState *state, ExprEvalStep *op, extern void ExecEvalRowNotNull(ExprState *state, ExprEvalStep *op,
ExprContext *econtext); ExprContext *econtext);
extern void ExecEvalArrayExpr(ExprState *state, ExprEvalStep *op); extern void ExecEvalArrayExpr(ExprState *state, ExprEvalStep *op);
extern void ExecEvalArrayCoerce(ExprState *state, ExprEvalStep *op); extern void ExecEvalArrayCoerce(ExprState *state, ExprEvalStep *op,
ExprContext *econtext);
extern void ExecEvalRow(ExprState *state, ExprEvalStep *op); extern void ExecEvalRow(ExprState *state, ExprEvalStep *op);
extern void ExecEvalMinMax(ExprState *state, ExprEvalStep *op); extern void ExecEvalMinMax(ExprState *state, ExprEvalStep *op);
extern void ExecEvalFieldSelect(ExprState *state, ExprEvalStep *op, extern void ExecEvalFieldSelect(ExprState *state, ExprEvalStep *op,
......
...@@ -820,11 +820,12 @@ typedef struct CoerceViaIO ...@@ -820,11 +820,12 @@ typedef struct CoerceViaIO
* ArrayCoerceExpr * ArrayCoerceExpr
* *
* ArrayCoerceExpr represents a type coercion from one array type to another, * ArrayCoerceExpr represents a type coercion from one array type to another,
* which is implemented by applying the indicated element-type coercion * which is implemented by applying the per-element coercion expression
* function to each element of the source array. If elemfuncid is InvalidOid * "elemexpr" to each element of the source array. Within elemexpr, the
* then the element types are binary-compatible, but the coercion still * source element is represented by a CaseTestExpr node. Note that even if
* requires some effort (we have to fix the element type ID stored in the * elemexpr is a no-op (that is, just CaseTestExpr + RelabelType), the
* array header). * coercion still requires some effort: we have to fix the element type OID
* stored in the array header.
* ---------------- * ----------------
*/ */
...@@ -832,11 +833,10 @@ typedef struct ArrayCoerceExpr ...@@ -832,11 +833,10 @@ typedef struct ArrayCoerceExpr
{ {
Expr xpr; Expr xpr;
Expr *arg; /* input expression (yields an array) */ Expr *arg; /* input expression (yields an array) */
Oid elemfuncid; /* OID of element coercion function, or 0 */ Expr *elemexpr; /* expression representing per-element work */
Oid resulttype; /* output type of coercion (an array type) */ Oid resulttype; /* output type of coercion (an array type) */
int32 resulttypmod; /* output typmod (also element typmod) */ int32 resulttypmod; /* output typmod (also element typmod) */
Oid resultcollid; /* OID of collation, or InvalidOid if none */ Oid resultcollid; /* OID of collation, or InvalidOid if none */
bool isExplicit; /* conversion semantics flag to pass to func */
CoercionForm coerceformat; /* how to display this node */ CoercionForm coerceformat; /* how to display this node */
int location; /* token location, or -1 if unknown */ int location; /* token location, or -1 if unknown */
} ArrayCoerceExpr; } ArrayCoerceExpr;
......
...@@ -48,9 +48,8 @@ extern Node *coerce_type(ParseState *pstate, Node *node, ...@@ -48,9 +48,8 @@ extern Node *coerce_type(ParseState *pstate, Node *node,
CoercionContext ccontext, CoercionForm cformat, int location); CoercionContext ccontext, CoercionForm cformat, int location);
extern Node *coerce_to_domain(Node *arg, Oid baseTypeId, int32 baseTypeMod, extern Node *coerce_to_domain(Node *arg, Oid baseTypeId, int32 baseTypeMod,
Oid typeId, Oid typeId,
CoercionForm cformat, int location, CoercionContext ccontext, CoercionForm cformat, int location,
bool hideInputCoercion, bool hideInputCoercion);
bool lengthCoercionDone);
extern Node *coerce_to_boolean(ParseState *pstate, Node *node, extern Node *coerce_to_boolean(ParseState *pstate, Node *node,
const char *constructName); const char *constructName);
......
...@@ -64,6 +64,10 @@ ...@@ -64,6 +64,10 @@
#include "fmgr.h" #include "fmgr.h"
#include "utils/expandeddatum.h" #include "utils/expandeddatum.h"
/* avoid including execnodes.h here */
struct ExprState;
struct ExprContext;
/* /*
* Arrays are varlena objects, so must meet the varlena convention that * Arrays are varlena objects, so must meet the varlena convention that
...@@ -360,8 +364,9 @@ extern ArrayType *array_set(ArrayType *array, int nSubscripts, int *indx, ...@@ -360,8 +364,9 @@ extern ArrayType *array_set(ArrayType *array, int nSubscripts, int *indx,
Datum dataValue, bool isNull, Datum dataValue, bool isNull,
int arraytyplen, int elmlen, bool elmbyval, char elmalign); int arraytyplen, int elmlen, bool elmbyval, char elmalign);
extern Datum array_map(FunctionCallInfo fcinfo, Oid retType, extern Datum array_map(Datum arrayd,
ArrayMapState *amstate); struct ExprState *exprstate, struct ExprContext *econtext,
Oid retType, ArrayMapState *amstate);
extern void array_bitmap_copy(bits8 *destbitmap, int destoffset, extern void array_bitmap_copy(bits8 *destbitmap, int destoffset,
const bits8 *srcbitmap, int srcoffset, const bits8 *srcbitmap, int srcoffset,
......
...@@ -310,6 +310,101 @@ Rules: ...@@ -310,6 +310,101 @@ Rules:
drop table dcomptable; drop table dcomptable;
drop type comptype cascade; drop type comptype cascade;
NOTICE: drop cascades to type dcomptypea NOTICE: drop cascades to type dcomptypea
-- Test arrays over domains
create domain posint as int check (value > 0);
create table pitable (f1 posint[]);
insert into pitable values(array[42]);
insert into pitable values(array[-1]); -- fail
ERROR: value for domain posint violates check constraint "posint_check"
insert into pitable values('{0}'); -- fail
ERROR: value for domain posint violates check constraint "posint_check"
LINE 1: insert into pitable values('{0}');
^
update pitable set f1[1] = f1[1] + 1;
update pitable set f1[1] = 0; -- fail
ERROR: value for domain posint violates check constraint "posint_check"
select * from pitable;
f1
------
{43}
(1 row)
drop table pitable;
create domain vc4 as varchar(4);
create table vc4table (f1 vc4[]);
insert into vc4table values(array['too long']); -- fail
ERROR: value too long for type character varying(4)
insert into vc4table values(array['too long']::vc4[]); -- cast truncates
select * from vc4table;
f1
----------
{"too "}
(1 row)
drop table vc4table;
drop type vc4;
-- You can sort of fake arrays-of-arrays by putting a domain in between
create domain dposinta as posint[];
create table dposintatable (f1 dposinta[]);
insert into dposintatable values(array[array[42]]); -- fail
ERROR: column "f1" is of type dposinta[] but expression is of type integer[]
LINE 1: insert into dposintatable values(array[array[42]]);
^
HINT: You will need to rewrite or cast the expression.
insert into dposintatable values(array[array[42]::posint[]]); -- still fail
ERROR: column "f1" is of type dposinta[] but expression is of type posint[]
LINE 1: insert into dposintatable values(array[array[42]::posint[]])...
^
HINT: You will need to rewrite or cast the expression.
insert into dposintatable values(array[array[42]::dposinta]); -- but this works
select f1, f1[1], (f1[1])[1] from dposintatable;
f1 | f1 | f1
----------+------+----
{"{42}"} | {42} | 42
(1 row)
select pg_typeof(f1) from dposintatable;
pg_typeof
------------
dposinta[]
(1 row)
select pg_typeof(f1[1]) from dposintatable;
pg_typeof
-----------
dposinta
(1 row)
select pg_typeof(f1[1][1]) from dposintatable;
pg_typeof
-----------
dposinta
(1 row)
select pg_typeof((f1[1])[1]) from dposintatable;
pg_typeof
-----------
posint
(1 row)
update dposintatable set f1[2] = array[99];
select f1, f1[1], (f1[2])[1] from dposintatable;
f1 | f1 | f1
-----------------+------+----
{"{42}","{99}"} | {42} | 99
(1 row)
-- it'd be nice if you could do something like this, but for now you can't:
update dposintatable set f1[2][1] = array[97];
ERROR: wrong number of array subscripts
-- maybe someday we can make this syntax work:
update dposintatable set (f1[2])[1] = array[98];
ERROR: syntax error at or near "["
LINE 1: update dposintatable set (f1[2])[1] = array[98];
^
drop table dposintatable;
drop domain posint cascade;
NOTICE: drop cascades to type dposinta
-- Test not-null restrictions -- Test not-null restrictions
create domain dnotnull varchar(15) NOT NULL; create domain dnotnull varchar(15) NOT NULL;
create domain dnull varchar(15); create domain dnull varchar(15);
......
...@@ -166,6 +166,49 @@ drop table dcomptable; ...@@ -166,6 +166,49 @@ drop table dcomptable;
drop type comptype cascade; drop type comptype cascade;
-- Test arrays over domains
create domain posint as int check (value > 0);
create table pitable (f1 posint[]);
insert into pitable values(array[42]);
insert into pitable values(array[-1]); -- fail
insert into pitable values('{0}'); -- fail
update pitable set f1[1] = f1[1] + 1;
update pitable set f1[1] = 0; -- fail
select * from pitable;
drop table pitable;
create domain vc4 as varchar(4);
create table vc4table (f1 vc4[]);
insert into vc4table values(array['too long']); -- fail
insert into vc4table values(array['too long']::vc4[]); -- cast truncates
select * from vc4table;
drop table vc4table;
drop type vc4;
-- You can sort of fake arrays-of-arrays by putting a domain in between
create domain dposinta as posint[];
create table dposintatable (f1 dposinta[]);
insert into dposintatable values(array[array[42]]); -- fail
insert into dposintatable values(array[array[42]::posint[]]); -- still fail
insert into dposintatable values(array[array[42]::dposinta]); -- but this works
select f1, f1[1], (f1[1])[1] from dposintatable;
select pg_typeof(f1) from dposintatable;
select pg_typeof(f1[1]) from dposintatable;
select pg_typeof(f1[1][1]) from dposintatable;
select pg_typeof((f1[1])[1]) from dposintatable;
update dposintatable set f1[2] = array[99];
select f1, f1[1], (f1[2])[1] from dposintatable;
-- it'd be nice if you could do something like this, but for now you can't:
update dposintatable set f1[2][1] = array[97];
-- maybe someday we can make this syntax work:
update dposintatable set (f1[2])[1] = array[98];
drop table dposintatable;
drop domain posint cascade;
-- Test not-null restrictions -- Test not-null restrictions
create domain dnotnull varchar(15) NOT NULL; create domain dnotnull varchar(15) NOT NULL;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment