Hi,

In <20240218015955.rmw5mcmobt5hb...@awork3.anarazel.de>
  "Re: Why is pq_begintypsend so slow?" on Sat, 17 Feb 2024 17:59:55 -0800,
  Andres Freund <and...@anarazel.de> wrote:

> v3-0008-wip-make-in-out-send-recv-calls-in-copy.c-cheaper.patch

It seems that this is an alternative approach of [1].

[1] 
https://www.postgresql.org/message-id/flat/20240215.153421.96888103784986803.kou%40clear-code.com#34df359e6d255795d16814ce138cc995


+typedef struct CopyInAttributeInfo
+{
+       AttrNumber      num;
+       const char *name;
+
+       Oid                     typioparam;
+       int32           typmod;
+
+       FmgrInfo        in_finfo;
+       union
+       {
+               FunctionCallInfoBaseData fcinfo;
+               char            fcinfo_data[SizeForFunctionCallInfo(3)];
+       }                       in_fcinfo;

Do we need one FunctionCallInfoBaseData for each attribute?
How about sharing one FunctionCallInfoBaseData by all
attributes like [1]?


@@ -956,20 +956,47 @@ NextCopyFrom(CopyFromState cstate, ExprContext *econtext,
 
                                values[m] = ExecEvalExpr(defexprs[m], econtext, 
&nulls[m]);
                        }
-
-                       /*
-                        * If ON_ERROR is specified with IGNORE, skip rows with 
soft
-                        * errors
-                        */
-                       else if (!InputFunctionCallSafe(&in_functions[m],
-                                                                               
        string,
-                                                                               
        typioparams[m],
-                                                                               
        att->atttypmod,
-                                                                               
        (Node *) cstate->escontext,
-                                                                               
        &values[m]))

Inlining InputFuncallCallSafe() here to use pre-initialized
fcinfo will decrease maintainability. Because we abstract
InputFunctionCall family in fmgr.c. If we define a
InputFunctionCall variant here, we need to change both of
fmgr.c and here when InputFunctionCall family is changed.
How about defining details in fmgr.c and call it here
instead like [1]?

+                               fcinfo->args[0].value = CStringGetDatum(string);
+                               fcinfo->args[0].isnull = false;
+                               fcinfo->args[1].value = 
ObjectIdGetDatum(attr->typioparam);
+                               fcinfo->args[1].isnull = false;
+                               fcinfo->args[2].value = 
Int32GetDatum(attr->typmod);
+                               fcinfo->args[2].isnull = false;

I think that "fcinfo->isnull = false;" is also needed like
[1].

@@ -1966,7 +1992,7 @@ CopyReadBinaryAttribute(CopyFromState cstate, FmgrInfo 
*flinfo,
        if (fld_size == -1)
        {
                *isnull = true;
-               return ReceiveFunctionCall(flinfo, NULL, typioparam, typmod);
+               return ReceiveFunctionCall(fcinfo->flinfo, NULL, 
attr->typioparam, attr->typmod);

Why pre-initialized fcinfo isn't used here?


Thanks,
-- 
kou


Reply via email to