8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-22 20:43:02 +01:00

Allow mixed (positional, named) arguments.

This commit is contained in:
Adriano dos Santos Fernandes 2023-05-02 21:58:44 -03:00
parent e452238cec
commit 5dbd5a46ef
7 changed files with 244 additions and 274 deletions

View File

@ -1,12 +1,15 @@
# Named arguments for function and procedure calling (FB 6.0) # Named arguments for function and procedure calling (FB 6.0)
Named arguments allows you to specify function and procedure arguments by their names, rather than only by their positions. Named arguments allows you to specify function and procedure arguments by their names, rather than only
by their positions.
It is especially useful when the routine have a lot of parameters and you want to specify them in arbitrary order or not specify some of them who have default values. It is especially useful when the routine have a lot of parameters and you want to specify them in arbitrary
order or not specify some of those who have default values.
As the positional syntax, all arguments without default values are required to be present in the call. As the positional syntax, all arguments without default values are required to be present in the call.
It's currently not possible to mix positional and named arguments in the same call. A call can use positional, named or mixed arguments. In mixed syntax, positional arguments must appear before
named arguments.
## Syntax ## Syntax
@ -23,8 +26,11 @@ It's currently not possible to mix positional and named arguments in the same ca
[RETURNING_VALUES ...] [RETURNING_VALUES ...]
<arguments> ::= <arguments> ::=
<named arguments> <positional arguments> |
<positional arguments> [ {<positional arguments>,} ] <named arguments>
<positional arguments> ::=
<value> [ {, <value>}... ]
<named arguments> ::= <named arguments> ::=
<named argument> [ {, <named argument>}... ] <named argument> [ {, <named argument>}... ]
@ -40,6 +46,11 @@ select function_name(parameter2 => 'Two', parameter1 => 1)
from rdb$database from rdb$database
``` ```
```
select function_name(1, parameter2 => 'Two')
from rdb$database
```
``` ```
execute procedure insert_customer( execute procedure insert_customer(
last_name => 'SCHUMACHER', last_name => 'SCHUMACHER',

View File

@ -12976,7 +12976,7 @@ DmlNode* UdfCallNode::parse(thread_db* tdbb, MemoryPool& pool, CompilerScratch*
predateCheck(node->function, "blr_invoke_function_type", "blr_invoke_function_args"); predateCheck(node->function, "blr_invoke_function_type", "blr_invoke_function_args");
argCount = blrReader.getWord(); argCount = blrReader.getWord();
node->args = PAR_args(tdbb, csb, argCount, (argNames ? argCount : MAX(argCount, node->function->fun_inputs))); node->args = PAR_args(tdbb, csb, argCount, MAX(argCount, node->function->fun_inputs));
break; break;
default: default:
@ -13013,6 +13013,14 @@ DmlNode* UdfCallNode::parse(thread_db* tdbb, MemoryPool& pool, CompilerScratch*
node->args = PAR_args(tdbb, csb, argCount, node->function->fun_inputs); node->args = PAR_args(tdbb, csb, argCount, node->function->fun_inputs);
} }
if (argNames && argNames->getCount() > argCount)
{
blrReader.setPos(argNamesPos);
PAR_error(csb,
Arg::Gds(isc_random) <<
"blr_invoke_function_arg_names count cannot be greater than blr_invoke_function_args");
}
if (!node->function) if (!node->function)
{ {
blrReader.setPos(startPos); blrReader.setPos(startPos);
@ -13038,91 +13046,71 @@ DmlNode* UdfCallNode::parse(thread_db* tdbb, MemoryPool& pool, CompilerScratch*
node->isSubRoutine = node->function->isSubRoutine(); node->isSubRoutine = node->function->isSubRoutine();
Arg::StatusVector mismatchStatus; Arg::StatusVector mismatchStatus;
mismatchStatus << Arg::Gds(isc_fun_param_mismatch) << name.toString();
const auto mismatchInitialLength = mismatchStatus.length();
if (!node->args) if (!node->args)
node->args = FB_NEW_POOL(pool) ValueListNode(pool); node->args = FB_NEW_POOL(pool) ValueListNode(pool);
if (argNames && argNames->getCount() != node->args->items.getCount()) const auto positionalArgCount = argNames ? argCount - argNames->getCount() : argCount;
auto argIt = node->args->items.begin();
LeftPooledMap<MetaName, NestConst<ValueExprNode>> argsByName;
if (positionalArgCount)
{ {
blrReader.setPos(argNamesPos); if (argCount > node->function->fun_inputs)
PAR_error(csb, mismatchStatus << Arg::Gds(isc_wronumarg);
Arg::Gds(isc_random) << "blr_invoke_function_arg_names count differs from blr_invoke_function_args");
for (auto pos = 0; pos < positionalArgCount; ++pos)
{
if (pos < node->function->fun_inputs)
{
const auto& parameter = node->function->getInputFields()[pos];
if (argsByName.put(parameter->prm_name, *argIt))
mismatchStatus << Arg::Gds(isc_param_multiple_assignments) << parameter->prm_name;
}
++argIt;
}
} }
if (argNames) if (argNames)
{ {
LeftPooledMap<MetaName, NestConst<ValueExprNode>> argsByName;
auto argIt = node->args->items.begin();
for (const auto& argName : *argNames) for (const auto& argName : *argNames)
{ {
if (argsByName.put(argName, *argIt++)) if (argsByName.put(argName, *argIt++))
mismatchStatus << Arg::Gds(isc_param_multiple_assignments) << argName; mismatchStatus << Arg::Gds(isc_param_multiple_assignments) << argName;
} }
node->args->items.resize(node->function->getInputFields().getCount());
argIt = node->args->items.begin();
for (auto& parameter : node->function->getInputFields())
{
if (const auto argValue = argsByName.get(parameter->prm_name))
{
*argIt = *argValue;
argsByName.remove(parameter->prm_name);
}
else
{
if (parameter->prm_default_value)
*argIt = CMP_clone_node(tdbb, csb, parameter->prm_default_value);
else
mismatchStatus << Arg::Gds(isc_param_no_default_not_specified) << parameter->prm_name;
}
++argIt;
}
if (argsByName.hasData())
{
for (const auto& argPair : argsByName)
mismatchStatus << Arg::Gds(isc_param_not_exist) << argPair.first;
}
} }
else
node->args->items.resize(node->function->getInputFields().getCount());
argIt = node->args->items.begin();
for (auto& parameter : node->function->getInputFields())
{ {
// Check to see if the argument count matches. if (const auto argValue = argsByName.get(parameter->prm_name))
if (argCount > node->function->fun_inputs)
mismatchStatus << Arg::Gds(isc_wronumarg);
else if (argCount < node->function->fun_inputs - node->function->getDefaultCount())
{ {
unsigned pos = 0; *argIt = *argValue;
argsByName.remove(parameter->prm_name);
for (auto& parameter : node->function->getInputFields())
{
if (++pos <= argCount)
continue;
mismatchStatus << Arg::Gds(isc_param_no_default_not_specified) << parameter->prm_name;
if (pos >= node->function->fun_inputs - node->function->getDefaultCount())
break;
}
} }
else else
{ {
for (unsigned pos = argCount; pos < node->function->getInputFields().getCount(); ++pos) if (parameter->prm_default_value)
{ *argIt = CMP_clone_node(tdbb, csb, parameter->prm_default_value);
auto parameter = node->function->getInputFields()[pos]; else
fb_assert(parameter->prm_default_value); mismatchStatus << Arg::Gds(isc_param_no_default_not_specified) << parameter->prm_name;
node->args->items[pos] = CMP_clone_node(tdbb, csb, parameter->prm_default_value);
}
} }
++argIt;
} }
if (mismatchStatus.length() > mismatchInitialLength) if (argsByName.hasData())
status_exception::raise(mismatchStatus); {
for (const auto& argPair : argsByName)
mismatchStatus << Arg::Gds(isc_param_not_exist) << argPair.first;
}
if (mismatchStatus.hasData())
status_exception::raise(Arg::Gds(isc_fun_param_mismatch) << name.toString() << mismatchStatus);
// CVC: I will track ufds only if a function is not being dropped. // CVC: I will track ufds only if a function is not being dropped.
if (!node->function->isSubRoutine() && csb->collectingDependencies()) if (!node->function->isSubRoutine() && csb->collectingDependencies())
@ -13589,20 +13577,31 @@ ValueExprNode* UdfCallNode::dsqlPass(DsqlCompilerScratch* dsqlScratch)
Arg::Gds(isc_random) << Arg::Str(name.toString())); Arg::Gds(isc_random) << Arg::Str(name.toString()));
} }
auto argIt = node->args->items.begin();
unsigned pos = 0;
while (pos < node->args->items.getCount() - (node->dsqlArgNames ? node->dsqlArgNames->getCount() : 0))
{
if (pos < node->dsqlFunction->udf_arguments.getCount())
{
PASS1_set_parameter_type(dsqlScratch, *argIt++,
[&] (dsc* desc) { *desc = node->dsqlFunction->udf_arguments[pos].desc; },
false);
}
++pos;
}
if (node->dsqlArgNames) if (node->dsqlArgNames)
{ {
fb_assert(node->dsqlArgNames->getCount() == node->args->items.getCount()); fb_assert(node->dsqlArgNames->getCount() <= node->args->items.getCount());
LeftPooledMap<MetaName, const dsc*> argsByName; LeftPooledMap<MetaName, const dsc*> argsByName;
for (const auto& arg : node->dsqlFunction->udf_arguments) for (const auto& arg : node->dsqlFunction->udf_arguments)
argsByName.put(arg.name, &arg.desc); argsByName.put(arg.name, &arg.desc);
bool mismatchError = false;
Arg::StatusVector mismatchStatus; Arg::StatusVector mismatchStatus;
mismatchStatus << Arg::Gds(isc_fun_param_mismatch) << Arg::Str(name.toString());
auto argIt = node->args->items.begin();
for (const auto& argName : *node->dsqlArgNames) for (const auto& argName : *node->dsqlArgNames)
{ {
@ -13613,32 +13612,13 @@ ValueExprNode* UdfCallNode::dsqlPass(DsqlCompilerScratch* dsqlScratch)
false); false);
} }
else else
{
mismatchError = true;
mismatchStatus << Arg::Gds(isc_param_not_exist) << argName; mismatchStatus << Arg::Gds(isc_param_not_exist) << argName;
}
++argIt; ++argIt;
} }
if (mismatchError) if (mismatchStatus.hasData())
status_exception::raise(mismatchStatus); status_exception::raise(Arg::Gds(isc_fun_param_mismatch) << name.toString() << mismatchStatus);
}
else
{
unsigned pos = 0;
for (auto& arg : node->args->items)
{
if (pos < node->dsqlFunction->udf_arguments.getCount())
{
PASS1_set_parameter_type(dsqlScratch, arg,
[&] (dsc* desc) { *desc = node->dsqlFunction->udf_arguments[pos].desc; },
false);
}
++pos;
}
} }
return node; return node;

View File

@ -3003,7 +3003,7 @@ DmlNode* ExecProcedureNode::parse(thread_db* tdbb, MemoryPool& pool, CompilerScr
predateCheck(node->procedure, "blr_invsel_procedure_type", "blr_invsel_procedure_in_args"); predateCheck(node->procedure, "blr_invsel_procedure_type", "blr_invsel_procedure_in_args");
inArgCount = blrReader.getWord(); inArgCount = blrReader.getWord();
node->inputSources = PAR_args(tdbb, csb, inArgCount, node->inputSources = PAR_args(tdbb, csb, inArgCount,
(inArgNames ? inArgCount : MAX(inArgCount, node->procedure->getInputFields().getCount()))); MAX(inArgCount, node->procedure->getInputFields().getCount()));
break; break;
case blr_invsel_procedure_out_arg_names: case blr_invsel_procedure_out_arg_names:
@ -3071,6 +3071,14 @@ DmlNode* ExecProcedureNode::parse(thread_db* tdbb, MemoryPool& pool, CompilerScr
break; break;
} }
if (inArgNames && inArgNames->getCount() > node->inputSources->items.getCount())
{
blrReader.setPos(inArgNamesPos);
PAR_error(csb,
Arg::Gds(isc_random) <<
"blr_invsel_procedure_in_arg_names count cannot be greater than blr_invsel_procedure_in_args");
}
if (!node->procedure) if (!node->procedure)
{ {
blrReader.setPos(blrStartPos); blrReader.setPos(blrStartPos);
@ -3092,14 +3100,6 @@ DmlNode* ExecProcedureNode::parse(thread_db* tdbb, MemoryPool& pool, CompilerScr
if (!node->outputTargets) if (!node->outputTargets)
node->outputTargets = FB_NEW_POOL(pool) ValueListNode(pool); node->outputTargets = FB_NEW_POOL(pool) ValueListNode(pool);
if (inArgNames && inArgNames->getCount() != node->inputSources->items.getCount())
{
blrReader.setPos(inArgNamesPos);
PAR_error(csb,
Arg::Gds(isc_random) <<
"blr_invsel_procedure_in_arg_names count differs from blr_invsel_procedure_in_args");
}
if (outArgNames && outArgNames->getCount() != node->outputTargets->items.getCount()) if (outArgNames && outArgNames->getCount() != node->outputTargets->items.getCount())
{ {
blrReader.setPos(outArgNamesPos); blrReader.setPos(outArgNamesPos);
@ -3125,13 +3125,9 @@ DmlNode* ExecProcedureNode::parse(thread_db* tdbb, MemoryPool& pool, CompilerScr
} }
} }
Arg::StatusVector mismatchStatus;
mismatchStatus << Arg::Gds(isc_prcmismat) << node->procedure->getName().toString();
const auto mismatchInitialLength = mismatchStatus.length();
node->inputTargets = FB_NEW_POOL(pool) ValueListNode(pool, node->procedure->getInputFields().getCount()); node->inputTargets = FB_NEW_POOL(pool) ValueListNode(pool, node->procedure->getInputFields().getCount());
mismatchStatus << CMP_procedure_arguments( Arg::StatusVector mismatchStatus = CMP_procedure_arguments(
tdbb, tdbb,
csb, csb,
node->procedure, node->procedure,
@ -3153,8 +3149,8 @@ DmlNode* ExecProcedureNode::parse(thread_db* tdbb, MemoryPool& pool, CompilerScr
node->outputSources, node->outputSources,
node->outputMessage); node->outputMessage);
if (mismatchStatus.length() > mismatchInitialLength) if (mismatchStatus.hasData())
status_exception::raise(mismatchStatus); status_exception::raise(Arg::Gds(isc_prcmismat) << node->procedure->getName().toString() << mismatchStatus);
if (csb->collectingDependencies() && !node->procedure->isSubRoutine()) if (csb->collectingDependencies() && !node->procedure->isSubRoutine())
{ {
@ -3246,63 +3242,59 @@ ExecProcedureNode* ExecProcedureNode::dsqlPass(DsqlCompilerScratch* dsqlScratch)
// Handle input parameters. // Handle input parameters.
if (node->dsqlInputArgNames) if (inputSources && inputSources->items.hasData())
{ {
fb_assert(node->dsqlInputArgNames->getCount() == node->inputSources->items.getCount()); auto sourceArgIt = node->inputSources->items.begin();
const auto sourceArgEnd = node->inputSources->items.end();
const auto positionalArgCount = node->inputSources->items.getCount() -
(node->dsqlInputArgNames ? node->dsqlInputArgNames->getCount() : 0);
const auto* field = procedure->prc_inputs;
unsigned pos = 0;
LeftPooledMap<MetaName, const dsql_fld*> argsByName; while (pos < positionalArgCount && field && sourceArgIt != sourceArgEnd)
for (const auto* field = procedure->prc_inputs; field; field = field->fld_next)
argsByName.put(field->fld_name, field);
bool mismatchError = false;
Arg::StatusVector mismatchStatus;
mismatchStatus << Arg::Gds(isc_prcmismat) << dsqlName.toString();
auto argIt = node->inputSources->items.begin();
for (const auto& argName : *node->dsqlInputArgNames)
{ {
if (const auto field = argsByName.get(argName)) dsc descNode;
{ DsqlDescMaker::fromField(&descNode, field);
dsc descNode;
DsqlDescMaker::fromField(&descNode, *field);
PASS1_set_parameter_type(dsqlScratch, *argIt, PASS1_set_parameter_type(dsqlScratch, *sourceArgIt,
[&] (dsc* desc) { *desc = descNode; }, [&] (dsc* desc) { *desc = descNode; },
false); false);
}
else
{
mismatchError = true;
mismatchStatus << Arg::Gds(isc_param_not_exist) << argName;
}
++argIt; field = field->fld_next;
++pos;
++sourceArgIt;
} }
if (mismatchError) if (node->dsqlInputArgNames)
status_exception::raise(mismatchStatus);
}
else
{
if (inputSources && inputSources->items.hasData())
{ {
// Initialize this stack variable, and make it look like a node. fb_assert(node->dsqlInputArgNames->getCount() <= node->inputSources->items.getCount());
dsc descNode;
auto ptr = node->inputSources->items.begin(); LeftPooledMap<MetaName, const dsql_fld*> argsByName;
const auto end = node->inputSources->items.end();
for (const dsql_fld* field = procedure->prc_inputs; field && ptr != end; ++ptr, field = field->fld_next) for (const auto* field = procedure->prc_inputs; field; field = field->fld_next)
argsByName.put(field->fld_name, field);
Arg::StatusVector mismatchStatus;
for (const auto& argName : *node->dsqlInputArgNames)
{ {
DEV_BLKCHK(field, dsql_type_fld); if (const auto field = argsByName.get(argName))
DEV_BLKCHK(*ptr, dsql_type_nod); {
DsqlDescMaker::fromField(&descNode, field); dsc descNode;
PASS1_set_parameter_type(dsqlScratch, *ptr, DsqlDescMaker::fromField(&descNode, *field);
[&] (dsc* desc) { *desc = descNode; },
false); PASS1_set_parameter_type(dsqlScratch, *sourceArgIt,
[&] (dsc* desc) { *desc = descNode; },
false);
}
else
mismatchStatus << Arg::Gds(isc_param_not_exist) << argName;
++sourceArgIt;
} }
if (mismatchStatus.hasData())
status_exception::raise(Arg::Gds(isc_prcmismat) << dsqlName.toString() << mismatchStatus);
} }
} }

View File

@ -8618,12 +8618,21 @@ argument_list_opt
%type <namedArguments> argument_list %type <namedArguments> argument_list
argument_list argument_list
: named_argument_list : named_argument_list
{ $$ = $1; }
| value_list | value_list
{ {
$$ = newNode<NonPooledPair<ObjectsArray<MetaName>*, ValueListNode*>>(); $$ = newNode<NonPooledPair<ObjectsArray<MetaName>*, ValueListNode*>>();
$$->second = $1; $$->second = $1;
} }
| value_list ',' named_argument_list
{
$$ = $3;
for (auto item : $$->second->items)
$1->add(item);
delete $$->second;
$$->second = $1;
}
; ;
%type <namedArguments> named_argument_list %type <namedArguments> named_argument_list

View File

@ -544,21 +544,38 @@ dsql_ctx* PASS1_make_context(DsqlCompilerScratch* dsqlScratch, RecordSourceNode*
if (count) if (count)
{ {
auto inputList = context->ctx_proc_inputs; auto inputList = context->ctx_proc_inputs;
auto argIt = inputList->items.begin();
const auto argEnd = inputList->items.end();
const auto positionalArgCount = inputList->items.getCount() -
(procNode->dsqlInputArgNames ? procNode->dsqlInputArgNames->getCount() : 0);
const auto* field = procedure->prc_inputs;
unsigned pos = 0;
while (pos < positionalArgCount && field && argIt != argEnd)
{
dsc descNode;
DsqlDescMaker::fromField(&descNode, field);
PASS1_set_parameter_type(dsqlScratch, *argIt,
[&] (dsc* desc) { *desc = descNode; },
false);
field = field->fld_next;
++pos;
++argIt;
}
if (procNode->dsqlInputArgNames) if (procNode->dsqlInputArgNames)
{ {
fb_assert(procNode->dsqlInputArgNames->getCount() == inputList->items.getCount()); fb_assert(procNode->dsqlInputArgNames->getCount() <= inputList->items.getCount());
LeftPooledMap<MetaName, const dsql_fld*> argsByName; LeftPooledMap<MetaName, const dsql_fld*> argsByName;
for (const auto* field = procedure->prc_inputs; field; field = field->fld_next) for (const auto* field = procedure->prc_inputs; field; field = field->fld_next)
argsByName.put(field->fld_name, field); argsByName.put(field->fld_name, field);
bool mismatchError = false;
Arg::StatusVector mismatchStatus; Arg::StatusVector mismatchStatus;
mismatchStatus << Arg::Gds(isc_prcmismat) << procNode->dsqlName.toString();
auto argIt = inputList->items.begin();
for (const auto& argName : *procNode->dsqlInputArgNames) for (const auto& argName : *procNode->dsqlInputArgNames)
{ {
@ -572,34 +589,13 @@ dsql_ctx* PASS1_make_context(DsqlCompilerScratch* dsqlScratch, RecordSourceNode*
false); false);
} }
else else
{
mismatchError = true;
mismatchStatus << Arg::Gds(isc_param_not_exist) << argName; mismatchStatus << Arg::Gds(isc_param_not_exist) << argName;
}
++argIt; ++argIt;
} }
if (mismatchError) if (mismatchStatus.hasData())
status_exception::raise(mismatchStatus); status_exception::raise(Arg::Gds(isc_prcmismat) << procNode->dsqlName.toString() << mismatchStatus);
}
else
{
// Initialize this stack variable, and make it look like a node
dsc descNode;
auto ptr = inputList->items.begin();
const auto end = inputList->items.end();
for (const dsql_fld* field = procedure->prc_inputs; ptr != end; ++ptr, field = field->fld_next)
{
DEV_BLKCHK(field, dsql_type_fld);
DEV_BLKCHK(*ptr, dsql_type_nod);
DsqlDescMaker::fromField(&descNode, field);
PASS1_set_parameter_type(dsqlScratch, *ptr,
[&] (dsc* desc) { *desc = descNode; },
false);
}
} }
} }
} }

View File

@ -987,7 +987,7 @@ ProcedureSourceNode* ProcedureSourceNode::parse(thread_db* tdbb, CompilerScratch
inArgCount = blrReader.getWord(); inArgCount = blrReader.getWord();
node->inputSources = PAR_args(tdbb, csb, inArgCount, node->inputSources = PAR_args(tdbb, csb, inArgCount,
(inArgNames ? inArgCount : MAX(inArgCount, node->procedure->getInputFields().getCount()))); MAX(inArgCount, node->procedure->getInputFields().getCount()));
break; break;
case blr_invsel_procedure_context: case blr_invsel_procedure_context:
@ -1069,6 +1069,14 @@ ProcedureSourceNode* ProcedureSourceNode::parse(thread_db* tdbb, CompilerScratch
fb_assert(false); fb_assert(false);
} }
if (inArgNames && inArgNames->getCount() > node->inputSources->items.getCount())
{
blrReader.setPos(inArgNamesPos);
PAR_error(csb,
Arg::Gds(isc_random) <<
"blr_invsel_procedure_in_arg_names count cannot be greater than blr_invsel_procedure_in_args");
}
if (!node->procedure) if (!node->procedure)
{ {
blrReader.setPos(blrStartPos); blrReader.setPos(blrStartPos);
@ -1119,21 +1127,9 @@ ProcedureSourceNode* ProcedureSourceNode::parse(thread_db* tdbb, CompilerScratch
if (!node->inputSources) if (!node->inputSources)
node->inputSources = FB_NEW_POOL(pool) ValueListNode(pool); node->inputSources = FB_NEW_POOL(pool) ValueListNode(pool);
if (inArgNames && inArgNames->getCount() != node->inputSources->items.getCount())
{
blrReader.setPos(inArgNamesPos);
PAR_error(csb,
Arg::Gds(isc_random) <<
"blr_invsel_procedure_in_arg_names count differs from blr_invsel_procedure_in_args");
}
Arg::StatusVector mismatchStatus;
mismatchStatus << Arg::Gds(isc_prcmismat) << node->procedure->getName().toString();
const auto mismatchInitialLength = mismatchStatus.length();
node->inputTargets = FB_NEW_POOL(pool) ValueListNode(pool, node->procedure->getInputFields().getCount()); node->inputTargets = FB_NEW_POOL(pool) ValueListNode(pool, node->procedure->getInputFields().getCount());
mismatchStatus << CMP_procedure_arguments( Arg::StatusVector mismatchStatus= CMP_procedure_arguments(
tdbb, tdbb,
csb, csb,
node->procedure, node->procedure,
@ -1144,8 +1140,8 @@ ProcedureSourceNode* ProcedureSourceNode::parse(thread_db* tdbb, CompilerScratch
node->inputTargets, node->inputTargets,
node->inputMessage); node->inputMessage);
if (mismatchStatus.length() > mismatchInitialLength) if (mismatchStatus.hasData())
status_exception::raise(mismatchStatus); status_exception::raise(Arg::Gds(isc_prcmismat) << node->procedure->getName().toString() << mismatchStatus);
if (csb->collectingDependencies() && !node->procedure->isSubRoutine()) if (csb->collectingDependencies() && !node->procedure->isSubRoutine())
{ {

View File

@ -484,9 +484,6 @@ Arg::StatusVector CMP_procedure_arguments(
else else
targets = FB_NEW_POOL(pool) ValueListNode(pool, argCount); targets = FB_NEW_POOL(pool) ValueListNode(pool, argCount);
auto sourceArgIt = sources->items.begin();
auto targetArgIt = targets->items.begin();
// We have a few parameters. Get on with creating the message block // We have a few parameters. Get on with creating the message block
// Outer messages map may start with 2, but they are always in the routine start. // Outer messages map may start with 2, but they are always in the routine start.
USHORT n = ++csb->csb_msg_number; USHORT n = ++csb->csb_msg_number;
@ -519,88 +516,77 @@ Arg::StatusVector CMP_procedure_arguments(
message->format = fmtCopy; message->format = fmtCopy;
// --- end of fix --- // --- end of fix ---
const auto positionalArgCount = argNames ? argCount - argNames->getCount() : argCount;
auto sourceArgIt = sources->items.begin();
LeftPooledMap<MetaName, NestConst<ValueExprNode>> argsByName;
if (positionalArgCount)
{
if (argCount > fields.getCount())
mismatchStatus << Arg::Gds(isc_wronumarg);
for (auto pos = 0; pos < positionalArgCount; ++pos)
{
if (pos < fields.getCount())
{
const auto& parameter = fields[pos];
if (argsByName.put(parameter->prm_name, *sourceArgIt))
mismatchStatus << Arg::Gds(isc_param_multiple_assignments) << parameter->prm_name;
}
++sourceArgIt;
}
}
if (argNames) if (argNames)
{ {
LeftPooledMap<MetaName, NestConst<ValueExprNode>> argsByName;
for (const auto& argName : *argNames) for (const auto& argName : *argNames)
{ {
if (argsByName.put(argName, *sourceArgIt++)) if (argsByName.put(argName, *sourceArgIt++))
mismatchStatus << Arg::Gds(isc_param_multiple_assignments) << argName; mismatchStatus << Arg::Gds(isc_param_multiple_assignments) << argName;
} }
sourceArgIt = sources->items.begin();
for (auto& parameter : fields)
{
if (const auto argValue = argsByName.get(parameter->prm_name))
{
*sourceArgIt = *argValue;
argsByName.remove(parameter->prm_name);
}
else if (isInput)
{
if (parameter->prm_default_value)
*sourceArgIt = CMP_clone_node(tdbb, csb, parameter->prm_default_value);
else
mismatchStatus << Arg::Gds(isc_param_no_default_not_specified) << parameter->prm_name;
}
++sourceArgIt;
const auto paramNode = FB_NEW_POOL(csb->csb_pool) ParameterNode(csb->csb_pool);
paramNode->messageNumber = message->messageNumber;
paramNode->message = message;
paramNode->argNumber = parameter->prm_number * 2;
const auto paramFlagNode = FB_NEW_POOL(csb->csb_pool) ParameterNode(csb->csb_pool);
paramFlagNode->messageNumber = message->messageNumber;
paramFlagNode->message = message;
paramFlagNode->argNumber = parameter->prm_number * 2 + 1;
paramNode->argFlag = paramFlagNode;
*targetArgIt++ = paramNode;
}
if (argsByName.hasData())
{
for (const auto& argPair : argsByName)
mismatchStatus << Arg::Gds(isc_param_not_exist) << argPair.first;
}
} }
else
sourceArgIt = sources->items.begin();
auto targetArgIt = targets->items.begin();
for (auto& parameter : fields)
{ {
for (unsigned i = 0; i < (isInput ? fields.getCount() : argCount); ++i) if (const auto argValue = argsByName.get(parameter->prm_name))
{ {
if (isInput) *sourceArgIt = *argValue;
{ argsByName.remove(parameter->prm_name);
// default value for parameter
if (i >= argCount)
{
auto parameter = fields[i];
if (parameter->prm_default_value)
*sourceArgIt = CMP_clone_node(tdbb, csb, parameter->prm_default_value);
}
++sourceArgIt;
}
const auto paramNode = FB_NEW_POOL(csb->csb_pool) ParameterNode(csb->csb_pool);
paramNode->messageNumber = message->messageNumber;
paramNode->message = message;
paramNode->argNumber = i * 2;
const auto paramFlagNode = FB_NEW_POOL(csb->csb_pool) ParameterNode(csb->csb_pool);
paramFlagNode->messageNumber = message->messageNumber;
paramFlagNode->message = message;
paramFlagNode->argNumber = i * 2 + 1;
paramNode->argFlag = paramFlagNode;
*targetArgIt++ = paramNode;
} }
else if (isInput)
{
if (parameter->prm_default_value)
*sourceArgIt = CMP_clone_node(tdbb, csb, parameter->prm_default_value);
else
mismatchStatus << Arg::Gds(isc_param_no_default_not_specified) << parameter->prm_name;
}
++sourceArgIt;
const auto paramNode = FB_NEW_POOL(csb->csb_pool) ParameterNode(csb->csb_pool);
paramNode->messageNumber = message->messageNumber;
paramNode->message = message;
paramNode->argNumber = parameter->prm_number * 2;
const auto paramFlagNode = FB_NEW_POOL(csb->csb_pool) ParameterNode(csb->csb_pool);
paramFlagNode->messageNumber = message->messageNumber;
paramFlagNode->message = message;
paramFlagNode->argNumber = parameter->prm_number * 2 + 1;
paramNode->argFlag = paramFlagNode;
*targetArgIt++ = paramNode;
}
if (argsByName.hasData())
{
for (const auto& argPair : argsByName)
mismatchStatus << Arg::Gds(isc_param_not_exist) << argPair.first;
} }
} }