| Index: runtime/vm/parser.cc
|
| diff --git a/runtime/vm/parser.cc b/runtime/vm/parser.cc
|
| index 6f639387c126ac232945dfea4ccc80cac1915855..014d39bf7d149750c66c48cb48949b10c0f3f913 100644
|
| --- a/runtime/vm/parser.cc
|
| +++ b/runtime/vm/parser.cc
|
| @@ -310,6 +310,8 @@ void ParsedFunction::AllocateVariables() {
|
| const intptr_t num_fixed_params = function().num_fixed_parameters();
|
| const intptr_t num_opt_params = function().NumOptionalParameters();
|
| const intptr_t num_params = num_fixed_params + num_opt_params;
|
| + const intptr_t type_args_slot = function().IsGeneric() ? 1 : 0;
|
| +
|
| // Compute start indices to parameters and locals, and the number of
|
| // parameters to copy.
|
| if (num_opt_params == 0) {
|
| @@ -330,8 +332,8 @@ void ParsedFunction::AllocateVariables() {
|
| // in the context(s).
|
| bool found_captured_variables = false;
|
| int next_free_frame_index = scope->AllocateVariables(
|
| - first_parameter_index_, num_params, first_stack_local_index_, NULL,
|
| - &found_captured_variables);
|
| + first_parameter_index_, num_params, type_args_slot,
|
| + first_stack_local_index_, NULL, &found_captured_variables);
|
|
|
| // Frame indices are relative to the frame pointer and are decreasing.
|
| ASSERT(next_free_frame_index <= first_stack_local_index_);
|
| @@ -1803,6 +1805,17 @@ void Parser::BuildDispatcherScope(const Function& func,
|
| // Build local scope for function and populate with the formal parameters.
|
| OpenFunctionBlock(func);
|
| AddFormalParamsToScope(¶ms, current_block_->scope);
|
| +
|
| + if (desc.TypeArgsLen() > 0) {
|
| + ASSERT(func.IsGeneric() && !func.HasGenericParent());
|
| + // Insert function type arguments variable to scope.
|
| + LocalVariable* type_args_var = new (Z) LocalVariable(
|
| + TokenPosition::kNoSource, TokenPosition::kNoSource,
|
| + Symbols::FunctionTypeArgumentsVar(), Object::dynamic_type());
|
| + current_block_->scope->AddVariable(type_args_var);
|
| + ASSERT(FunctionLevel() == 0);
|
| + parsed_function_->set_function_type_arguments(type_args_var);
|
| + }
|
| }
|
|
|
|
|
| @@ -1822,7 +1835,9 @@ SequenceNode* Parser::ParseNoSuchMethodDispatcher(const Function& func) {
|
|
|
| // Receiver is local 0.
|
| LocalScope* scope = current_block_->scope;
|
| - ArgumentListNode* func_args = new ArgumentListNode(token_pos);
|
| + ArgumentListNode* func_args = new ArgumentListNode(
|
| + token_pos, parsed_function_->function_type_arguments(),
|
| + desc.TypeArgsLen());
|
| for (intptr_t i = 0; i < desc.Count(); ++i) {
|
| func_args->Add(new LoadLocalNode(token_pos, scope->VariableAt(i)));
|
| }
|
| @@ -1898,9 +1913,12 @@ SequenceNode* Parser::ParseInvokeFieldDispatcher(const Function& func) {
|
| }
|
|
|
| // Pass arguments 1..n to the closure call.
|
| - ArgumentListNode* args = new (Z) ArgumentListNode(token_pos);
|
| + ArgumentListNode* args = new (Z)
|
| + ArgumentListNode(token_pos, parsed_function_->function_type_arguments(),
|
| + desc.TypeArgsLen());
|
| const Array& names =
|
| Array::Handle(Z, Array::New(desc.NamedCount(), Heap::kOld));
|
| +
|
| // Positional parameters.
|
| intptr_t i = 1;
|
| for (; i < desc.PositionalCount(); ++i) {
|
| @@ -5698,7 +5716,11 @@ RawTypeArguments* Parser::ParseTypeArguments(
|
| ReportError("right angle bracket expected");
|
| }
|
| if (finalization != ClassFinalizer::kIgnore) {
|
| - return NewTypeArguments(types);
|
| + TypeArguments& type_args = TypeArguments::Handle(NewTypeArguments(types));
|
| + if (finalization == ClassFinalizer::kCanonicalize) {
|
| + type_args = type_args.Canonicalize();
|
| + }
|
| + return type_args.raw();
|
| }
|
| }
|
| return TypeArguments::null();
|
| @@ -12565,12 +12587,15 @@ void Parser::ResolveType(AbstractType* type) {
|
| if (type->arguments() != TypeArguments::null()) {
|
| const TypeArguments& arguments =
|
| TypeArguments::Handle(Z, type->arguments());
|
| - const intptr_t num_arguments = arguments.Length();
|
| - AbstractType& type_argument = AbstractType::Handle(Z);
|
| - for (intptr_t i = 0; i < num_arguments; i++) {
|
| - type_argument = arguments.TypeAt(i);
|
| - ResolveType(&type_argument);
|
| - arguments.SetTypeAt(i, type_argument);
|
| + // Already resolved if canonical.
|
| + if (!arguments.IsCanonical()) {
|
| + const intptr_t num_arguments = arguments.Length();
|
| + AbstractType& type_argument = AbstractType::Handle(Z);
|
| + for (intptr_t i = 0; i < num_arguments; i++) {
|
| + type_argument = arguments.TypeAt(i);
|
| + ResolveType(&type_argument);
|
| + arguments.SetTypeAt(i, type_argument);
|
| + }
|
| }
|
| }
|
| if (type->IsFunctionType()) {
|
|
|