Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(150)

Unified Diff: sdk/lib/_internal/compiler/implementation/types/concrete_types_inferrer.dart

Issue 11863007: Fix optional arguments handling. Handling of default values is for a further (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: sync to head Created 7 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « no previous file | tests/compiler/dart2js/cpa_inference_test.dart » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: sdk/lib/_internal/compiler/implementation/types/concrete_types_inferrer.dart
diff --git a/sdk/lib/_internal/compiler/implementation/types/concrete_types_inferrer.dart b/sdk/lib/_internal/compiler/implementation/types/concrete_types_inferrer.dart
index eb5f364188e21a6cd654cc8ea8a9bbfde5b78dc4..c76d91b13086939b4acccbf5b7eac51e45a023de 100644
--- a/sdk/lib/_internal/compiler/implementation/types/concrete_types_inferrer.dart
+++ b/sdk/lib/_internal/compiler/implementation/types/concrete_types_inferrer.dart
@@ -675,62 +675,81 @@ class ConcreteTypesInferrer {
ArgumentsTypes argumentsTypes) {
final Map<Element, ConcreteType> result = new Map<Element, ConcreteType>();
final FunctionSignature signature = function.computeSignature(compiler);
- // too many arguments
+
+ // guard 1: too many arguments
if (argumentsTypes.length > signature.parameterCount) {
return null;
}
- // not enough arguments
+ // guard 2: not enough arguments
if (argumentsTypes.positional.length < signature.requiredParameterCount) {
return null;
}
- final HasNextIterator<ConcreteType> remainingPositionalArguments =
- new HasNextIterator<ConcreteType>(argumentsTypes.positional.iterator);
+ // guard 3: too many positional arguments
+ if (signature.optionalParametersAreNamed &&
+ argumentsTypes.positional.length > signature.requiredParameterCount) {
+ return null;
+ }
+
+ handleLeftoverOptionalParameter(Element parameter) {
+ // TODO(polux): use default value whenever available
+ // TODO(polux): add a marker to indicate whether an argument was provided
+ // in order to handle "?parameter" tests
+ result[parameter] = singletonConcreteType(const NullBaseType());
+ }
+
+ final Iterator<ConcreteType> remainingPositionalArguments =
+ argumentsTypes.positional.iterator;
// we attach each positional parameter to its corresponding positional
// argument
for (Link<Element> requiredParameters = signature.requiredParameters;
- !requiredParameters.isEmpty;
- requiredParameters = requiredParameters.tail) {
+ !requiredParameters.isEmpty;
+ requiredParameters = requiredParameters.tail) {
final Element requiredParameter = requiredParameters.head;
- // we know next() is defined because of the guard above
- result[requiredParameter] = remainingPositionalArguments.next();
+ // we know moveNext() succeeds because of guard 2
+ remainingPositionalArguments.moveNext();
+ result[requiredParameter] = remainingPositionalArguments.current;
}
- // we attach the remaining positional arguments to their corresponding
- // named arguments
- Link<Element> remainingNamedParameters = signature.optionalParameters;
- while (remainingPositionalArguments.hasNext) {
- final Element namedParameter = remainingNamedParameters.head;
- result[namedParameter] = remainingPositionalArguments.next();
- // we know tail is defined because of the guard above
- remainingNamedParameters = remainingNamedParameters.tail;
- }
- // we build a map out of the remaining named parameters
- final Map<SourceString, Element> leftOverNamedParameters =
- new Map<SourceString, Element>();
- for (;
- !remainingNamedParameters.isEmpty;
- remainingNamedParameters = remainingNamedParameters.tail) {
- final Element namedParameter = remainingNamedParameters.head;
- leftOverNamedParameters[namedParameter.name] = namedParameter;
+ if (signature.optionalParametersAreNamed) {
+ // we build a map out of the remaining named parameters
+ Link<Element> remainingOptionalParameters = signature.optionalParameters;
+ final Map<SourceString, Element> leftOverNamedParameters =
+ new Map<SourceString, Element>();
+ for (;
+ !remainingOptionalParameters.isEmpty;
+ remainingOptionalParameters = remainingOptionalParameters.tail) {
+ final Element namedParameter = remainingOptionalParameters.head;
+ leftOverNamedParameters[namedParameter.name] = namedParameter;
+ }
+ // we attach the named arguments to their corresponding optional
+ // parameters
+ for (Identifier identifier in argumentsTypes.named.keys) {
+ final ConcreteType concreteType = argumentsTypes.named[identifier];
+ SourceString source = identifier.source;
+ final Element namedParameter = leftOverNamedParameters[source];
+ // unexisting or already used named parameter
+ if (namedParameter == null) return null;
+ result[namedParameter] = concreteType;
+ leftOverNamedParameters.remove(source);
+ }
+ leftOverNamedParameters.forEach((_, Element parameter) {
+ handleLeftoverOptionalParameter(parameter);
+ });
+ } else { // optional parameters are positional
+ // we attach the remaining positional arguments to their corresponding
+ // optional parameters
+ Link<Element> remainingOptionalParameters = signature.optionalParameters;
+ while (remainingPositionalArguments.moveNext()) {
+ final Element optionalParameter = remainingOptionalParameters.head;
+ result[optionalParameter] = remainingPositionalArguments.current;
+ // we know tail is defined because of guard 1
+ remainingOptionalParameters = remainingOptionalParameters.tail;
+ }
+ for (;
+ !remainingOptionalParameters.isEmpty;
+ remainingOptionalParameters = remainingOptionalParameters.tail) {
+ handleLeftoverOptionalParameter(remainingOptionalParameters.head);
+ }
}
- // we attach the named arguments to their corresponding named paramaters
- // (we don't use foreach because we want to be able to return early)
- for (Identifier identifier in argumentsTypes.named.keys) {
- final ConcreteType concreteType = argumentsTypes.named[identifier];
- SourceString source = identifier.source;
- final Element namedParameter = leftOverNamedParameters[source];
- // unexisting or already used named parameter
- if (namedParameter == null) return null;
- result[namedParameter] = concreteType;
- leftOverNamedParameters.remove(source);
- };
- // we use null for each unused named parameter
- // TODO(polux): use default value whenever available
- // TODO(polux): add a marker to indicate whether an argument was provided
- // in order to handle "?parameter" tests
- leftOverNamedParameters.forEach((_, Element namedParameter) {
- result[namedParameter] =
- singletonConcreteType(const NullBaseType());
- });
return result;
}
« no previous file with comments | « no previous file | tests/compiler/dart2js/cpa_inference_test.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698