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

Side by Side Diff: vm/flow_graph_compiler_x64.cc

Issue 10535180: Allow implicit 'close your eyes' of native methods. This will hopefully fix issue 3466. (Closed) Base URL: http://dart.googlecode.com/svn/branches/bleeding_edge/dart/runtime/
Patch Set: Created 8 years, 6 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 unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
2 // for details. All rights reserved. Use of this source code is governed by a 2 // for details. All rights reserved. Use of this source code is governed by a
3 // BSD-style license that can be found in the LICENSE file. 3 // BSD-style license that can be found in the LICENSE file.
4 4
5 #include "vm/globals.h" // Needed here to get TARGET_ARCH_X64. 5 #include "vm/globals.h" // Needed here to get TARGET_ARCH_X64.
6 #if defined(TARGET_ARCH_X64) 6 #if defined(TARGET_ARCH_X64)
7 7
8 #include "vm/flow_graph_compiler.h" 8 #include "vm/flow_graph_compiler.h"
9 9
10 #include "lib/error.h" 10 #include "lib/error.h"
(...skipping 630 matching lines...) Expand 10 before | Expand all | Expand 10 after
641 Location loc = locs->in(i); 641 Location loc = locs->in(i);
642 ASSERT(loc.kind() == Location::kRegister); 642 ASSERT(loc.kind() == Location::kRegister);
643 __ popq(loc.reg()); 643 __ popq(loc.reg());
644 } 644 }
645 } 645 }
646 646
647 647
648 // Copied from CodeGenerator::CopyParameters (CodeGenerator will be deprecated). 648 // Copied from CodeGenerator::CopyParameters (CodeGenerator will be deprecated).
649 void FlowGraphCompiler::CopyParameters() { 649 void FlowGraphCompiler::CopyParameters() {
650 const Function& function = parsed_function().function(); 650 const Function& function = parsed_function().function();
651 const bool is_implicit_native_closure =
652 function.is_native() && function.IsImplicitClosureFunction();
regis 2012/06/15 17:13:34 ditto
siva 2012/06/16 00:25:43 Done.
651 LocalScope* scope = parsed_function().node_sequence()->scope(); 653 LocalScope* scope = parsed_function().node_sequence()->scope();
652 const int num_fixed_params = function.num_fixed_parameters(); 654 const int num_fixed_params = function.num_fixed_parameters();
653 const int num_opt_params = function.num_optional_parameters(); 655 const int num_opt_params = function.num_optional_parameters();
654 ASSERT(parsed_function().first_parameter_index() == 656 ASSERT(parsed_function().first_parameter_index() ==
655 ParsedFunction::kFirstLocalSlotIndex); 657 ParsedFunction::kFirstLocalSlotIndex);
656 // Copy positional arguments. 658 // Copy positional arguments.
657 // Check that no fewer than num_fixed_params positional arguments are passed 659 // Check that no fewer than num_fixed_params positional arguments are passed
658 // in and that no more than num_params arguments are passed in. 660 // in and that no more than num_params arguments are passed in.
659 // Passed argument i at fp[1 + argc - i] 661 // Passed argument i at fp[1 + argc - i]
660 // copied to fp[ParsedFunction::kFirstLocalSlotIndex - i]. 662 // copied to fp[ParsedFunction::kFirstLocalSlotIndex - i].
661 const int num_params = num_fixed_params + num_opt_params; 663 const int num_params = num_fixed_params + num_opt_params;
662 664
663 // Total number of args is the first Smi in args descriptor array (R10). 665 // Total number of args is the first Smi in args descriptor array (R10).
664 __ movq(RBX, FieldAddress(R10, Array::data_offset())); 666 __ movq(RBX, FieldAddress(R10, Array::data_offset()));
665 // Check that num_args <= num_params. 667 // Check that num_args <= num_params.
666 Label wrong_num_arguments; 668 Label wrong_num_arguments;
667 __ cmpq(RBX, Immediate(Smi::RawValue(num_params))); 669 __ cmpq(RBX, Immediate(Smi::RawValue(num_params)));
668 __ j(GREATER, &wrong_num_arguments); 670 __ j(GREATER, &wrong_num_arguments);
669 // Number of positional args is the second Smi in descriptor array (R10). 671 // Number of positional args is the second Smi in descriptor array (R10).
670 __ movq(RCX, FieldAddress(R10, Array::data_offset() + (1 * kWordSize))); 672 __ movq(RCX, FieldAddress(R10, Array::data_offset() + (1 * kWordSize)));
671 // Check that num_pos_args >= num_fixed_params. 673 // Check that num_pos_args >= num_fixed_params.
672 __ cmpq(RCX, Immediate(Smi::RawValue(num_fixed_params))); 674 __ cmpq(RCX, Immediate(Smi::RawValue(num_fixed_params)));
673 __ j(LESS, &wrong_num_arguments); 675 __ j(LESS, &wrong_num_arguments);
676
674 // Since RBX and RCX are Smi, use TIMES_4 instead of TIMES_8. 677 // Since RBX and RCX are Smi, use TIMES_4 instead of TIMES_8.
675 // Let RBX point to the last passed positional argument, i.e. to 678 // Let RBX point to the last passed positional argument, i.e. to
676 // fp[1 + num_args - (num_pos_args - 1)]. 679 // fp[1 + num_args - (num_pos_args - 1)].
677 __ subq(RBX, RCX); 680 __ subq(RBX, RCX);
678 __ leaq(RBX, Address(RBP, RBX, TIMES_4, 2 * kWordSize)); 681 __ leaq(RBX, Address(RBP, RBX, TIMES_4, 2 * kWordSize));
679 // Let RDI point to the last copied positional argument, i.e. to 682 // Let RDI point to the last copied positional argument, i.e. to
680 // fp[ParsedFunction::kFirstLocalSlotIndex - (num_pos_args - 1)]. 683 // fp[ParsedFunction::kFirstLocalSlotIndex - (num_pos_args - 1)].
684 int implicit_this_param_pos = is_implicit_native_closure ? -1 : 0;
685 const int index =
686 ParsedFunction::kFirstLocalSlotIndex + 1 + implicit_this_param_pos;
687 // First copy captured receiver if function is an implicit native closure.
688 if (is_implicit_native_closure) {
689 __ movq(RAX, FieldAddress(CTX, Context::variable_offset(0)));
690 __ movq(Address(RBP, (index * kWordSize)), RAX);
691 }
681 __ SmiUntag(RCX); 692 __ SmiUntag(RCX);
682 __ movq(RAX, RCX); 693 __ movq(RAX, RCX);
683 __ negq(RAX); 694 __ negq(RAX);
684 const int index = ParsedFunction::kFirstLocalSlotIndex + 1;
685 // -num_pos_args is in RAX. 695 // -num_pos_args is in RAX.
686 // (ParsedFunction::kFirstLocalSlotIndex + 1) is in index. 696 // (ParsedFunction::kFirstLocalSlotIndex + 1 + implicit_this_param_pos)
697 // is in index.
687 __ leaq(RDI, Address(RBP, RAX, TIMES_8, (index * kWordSize))); 698 __ leaq(RDI, Address(RBP, RAX, TIMES_8, (index * kWordSize)));
688 Label loop, loop_condition; 699 Label loop, loop_condition;
689 __ jmp(&loop_condition, Assembler::kNearJump); 700 __ jmp(&loop_condition, Assembler::kNearJump);
690 // We do not use the final allocation index of the variable here, i.e. 701 // We do not use the final allocation index of the variable here, i.e.
691 // scope->VariableAt(i)->index(), because captured variables still need 702 // scope->VariableAt(i)->index(), because captured variables still need
692 // to be copied to the context that is not yet allocated. 703 // to be copied to the context that is not yet allocated.
693 const Address argument_addr(RBX, RCX, TIMES_8, 0); 704 const Address argument_addr(RBX, RCX, TIMES_8, 0);
694 const Address copy_addr(RDI, RCX, TIMES_8, 0); 705 const Address copy_addr(RDI, RCX, TIMES_8, 0);
695 __ Bind(&loop); 706 __ Bind(&loop);
696 __ movq(RAX, argument_addr); 707 __ movq(RAX, argument_addr);
697 __ movq(copy_addr, RAX); 708 __ movq(copy_addr, RAX);
698 __ Bind(&loop_condition); 709 __ Bind(&loop_condition);
699 __ decq(RCX); 710 __ decq(RCX);
700 __ j(POSITIVE, &loop, Assembler::kNearJump); 711 __ j(POSITIVE, &loop, Assembler::kNearJump);
701 712
702 // Copy or initialize optional named arguments. 713 // Copy or initialize optional named arguments.
703 ASSERT(num_opt_params > 0); // Or we would not have to copy arguments.
704 // Start by alphabetically sorting the names of the optional parameters.
705 LocalVariable** opt_param = new LocalVariable*[num_opt_params];
706 int* opt_param_position = new int[num_opt_params];
707 for (int pos = num_fixed_params; pos < num_params; pos++) {
708 LocalVariable* parameter = scope->VariableAt(pos);
709 const String& opt_param_name = parameter->name();
710 int i = pos - num_fixed_params;
711 while (--i >= 0) {
712 LocalVariable* param_i = opt_param[i];
713 const intptr_t result = opt_param_name.CompareTo(param_i->name());
714 ASSERT(result != 0);
715 if (result > 0) break;
716 opt_param[i + 1] = opt_param[i];
717 opt_param_position[i + 1] = opt_param_position[i];
718 }
719 opt_param[i + 1] = parameter;
720 opt_param_position[i + 1] = pos;
721 }
722 // Generate code handling each optional parameter in alphabetical order.
723 // Total number of args is the first Smi in args descriptor array (R10).
724 __ movq(RBX, FieldAddress(R10, Array::data_offset()));
725 // Number of positional args is the second Smi in descriptor array (R10).
726 __ movq(RCX, FieldAddress(R10, Array::data_offset() + (1 * kWordSize)));
727 __ SmiUntag(RCX);
728 // Let RBX point to the first passed argument, i.e. to fp[1 + argc - 0].
729 __ leaq(RBX, Address(RBP, RBX, TIMES_4, kWordSize)); // RBX is Smi.
730 // Let EDI point to the name/pos pair of the first named argument.
731 __ leaq(RDI, FieldAddress(R10, Array::data_offset() + (2 * kWordSize)));
732 for (int i = 0; i < num_opt_params; i++) {
733 // Handle this optional parameter only if k or fewer positional arguments
734 // have been passed, where k is the position of this optional parameter in
735 // the formal parameter list.
736 Label load_default_value, assign_optional_parameter, next_parameter;
737 const int param_pos = opt_param_position[i];
738 __ cmpq(RCX, Immediate(param_pos));
739 __ j(GREATER, &next_parameter, Assembler::kNearJump);
740 // Check if this named parameter was passed in.
741 __ movq(RAX, Address(RDI, 0)); // Load RAX with the name of the argument.
742 __ CompareObject(RAX, opt_param[i]->name());
743 __ j(NOT_EQUAL, &load_default_value, Assembler::kNearJump);
744 // Load RAX with passed-in argument at provided arg_pos, i.e. at
745 // fp[1 + argc - arg_pos].
746 __ movq(RAX, Address(RDI, kWordSize)); // RAX is arg_pos as Smi.
747 __ addq(RDI, Immediate(2 * kWordSize)); // Point to next name/pos pair.
748 __ negq(RAX);
749 Address argument_addr(RBX, RAX, TIMES_4, 0); // RAX is a negative Smi.
750 __ movq(RAX, argument_addr);
751 __ jmp(&assign_optional_parameter, Assembler::kNearJump);
752 __ Bind(&load_default_value);
753 // Load RAX with default argument at pos.
754 const Object& value = Object::ZoneHandle(
755 parsed_function().default_parameter_values().At(
756 param_pos - num_fixed_params));
757 __ LoadObject(RAX, value);
758 __ Bind(&assign_optional_parameter);
759 // Assign RAX to fp[ParsedFunction::kFirstLocalSlotIndex - param_pos].
760 // We do not use the final allocation index of the variable here, i.e.
761 // scope->VariableAt(i)->index(), because captured variables still need
762 // to be copied to the context that is not yet allocated.
763 const Address param_addr(
764 RBP, (ParsedFunction::kFirstLocalSlotIndex - param_pos) * kWordSize);
765 __ movq(param_addr, RAX);
766 __ Bind(&next_parameter);
767 }
768 delete[] opt_param;
769 delete[] opt_param_position;
770 // Check that RDI now points to the null terminator in the array descriptor.
771 const Immediate raw_null = 714 const Immediate raw_null =
772 Immediate(reinterpret_cast<intptr_t>(Object::null())); 715 Immediate(reinterpret_cast<intptr_t>(Object::null()));
773 Label all_arguments_processed; 716 Label all_arguments_processed;
774 __ cmpq(Address(RDI, 0), raw_null); 717 if (num_opt_params > 0) {
775 __ j(EQUAL, &all_arguments_processed, Assembler::kNearJump); 718 // Start by alphabetically sorting the names of the optional parameters.
719 LocalVariable** opt_param = new LocalVariable*[num_opt_params];
720 int* opt_param_position = new int[num_opt_params];
721 for (int pos = num_fixed_params; pos < num_params; pos++) {
722 LocalVariable* parameter = scope->VariableAt(pos);
723 const String& opt_param_name = parameter->name();
724 int i = pos - num_fixed_params;
725 while (--i >= 0) {
726 LocalVariable* param_i = opt_param[i];
727 const intptr_t result = opt_param_name.CompareTo(param_i->name());
728 ASSERT(result != 0);
729 if (result > 0) break;
730 opt_param[i + 1] = opt_param[i];
731 opt_param_position[i + 1] = opt_param_position[i];
732 }
733 opt_param[i + 1] = parameter;
734 opt_param_position[i + 1] = pos;
735 }
736 // Generate code handling each optional parameter in alphabetical order.
737 // Total number of args is the first Smi in args descriptor array (R10).
738 __ movq(RBX, FieldAddress(R10, Array::data_offset()));
739 // Number of positional args is the second Smi in descriptor array (R10).
740 __ movq(RCX, FieldAddress(R10, Array::data_offset() + (1 * kWordSize)));
741 __ SmiUntag(RCX);
742 // Let RBX point to the first passed argument, i.e. to fp[1 + argc - 0].
743 __ leaq(RBX, Address(RBP, RBX, TIMES_4, kWordSize)); // RBX is Smi.
744 // Let EDI point to the name/pos pair of the first named argument.
745 __ leaq(RDI, FieldAddress(R10, Array::data_offset() + (2 * kWordSize)));
746 for (int i = 0; i < num_opt_params; i++) {
747 // Handle this optional parameter only if k or fewer positional arguments
748 // have been passed, where k is the position of this optional parameter in
749 // the formal parameter list.
750 Label load_default_value, assign_optional_parameter, next_parameter;
751 const int param_pos = opt_param_position[i];
752 __ cmpq(RCX, Immediate(param_pos));
753 __ j(GREATER, &next_parameter, Assembler::kNearJump);
754 // Check if this named parameter was passed in.
755 __ movq(RAX, Address(RDI, 0)); // Load RAX with the name of the argument.
756 __ CompareObject(RAX, opt_param[i]->name());
757 __ j(NOT_EQUAL, &load_default_value, Assembler::kNearJump);
758 // Load RAX with passed-in argument at provided arg_pos, i.e. at
759 // fp[1 + argc - arg_pos].
760 __ movq(RAX, Address(RDI, kWordSize)); // RAX is arg_pos as Smi.
761 __ addq(RDI, Immediate(2 * kWordSize)); // Point to next name/pos pair.
762 __ negq(RAX);
763 Address argument_addr(RBX, RAX, TIMES_4, 0); // RAX is a negative Smi.
764 __ movq(RAX, argument_addr);
765 __ jmp(&assign_optional_parameter, Assembler::kNearJump);
766 __ Bind(&load_default_value);
767 // Load RAX with default argument at pos.
768 const Object& value = Object::ZoneHandle(
769 parsed_function().default_parameter_values().At(
770 param_pos - num_fixed_params));
771 __ LoadObject(RAX, value);
772 __ Bind(&assign_optional_parameter);
773 // Assign RAX to fp[ParsedFunction::kFirstLocalSlotIndex - param_pos].
774 // We do not use the final allocation index of the variable here, i.e.
775 // scope->VariableAt(i)->index(), because captured variables still need
776 // to be copied to the context that is not yet allocated.
777 intptr_t computed_param_pos = (ParsedFunction::kFirstLocalSlotIndex -
778 param_pos + implicit_this_param_pos);
779 const Address param_addr(RBP, (computed_param_pos * kWordSize));
780 __ movq(param_addr, RAX);
781 __ Bind(&next_parameter);
782 }
783 delete[] opt_param;
784 delete[] opt_param_position;
785 // Check that RDI now points to the null terminator in the array descriptor.
786 __ cmpq(Address(RDI, 0), raw_null);
787 __ j(EQUAL, &all_arguments_processed, Assembler::kNearJump);
788 } else {
789 ASSERT(is_implicit_native_closure);
790 __ jmp(&all_arguments_processed, Assembler::kNearJump);
791 }
776 792
777 __ Bind(&wrong_num_arguments); 793 __ Bind(&wrong_num_arguments);
778 if (StackSize() != 0) { 794 if (StackSize() != 0) {
779 // We need to unwind the space we reserved for locals and copied parameters. 795 // We need to unwind the space we reserved for locals and copied parameters.
780 // The NoSuchMethodFunction stub does not expect to see that area on the 796 // The NoSuchMethodFunction stub does not expect to see that area on the
781 // stack. 797 // stack.
782 __ addq(RSP, Immediate(StackSize() * kWordSize)); 798 __ addq(RSP, Immediate(StackSize() * kWordSize));
783 } 799 }
784 if (function.IsClosureFunction()) { 800 if (function.IsClosureFunction()) {
785 GenerateCallRuntime(AstNode::kNoId, 801 GenerateCallRuntime(AstNode::kNoId,
(...skipping 240 matching lines...) Expand 10 before | Expand all | Expand 10 after
1026 __ cvtsi2sd(result, temp); 1042 __ cvtsi2sd(result, temp);
1027 __ Bind(&done); 1043 __ Bind(&done);
1028 } 1044 }
1029 1045
1030 1046
1031 #undef __ 1047 #undef __
1032 1048
1033 } // namespace dart 1049 } // namespace dart
1034 1050
1035 #endif // defined TARGET_ARCH_X64 1051 #endif // defined TARGET_ARCH_X64
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698