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

Side by Side Diff: runtime/vm/intrinsifier.cc

Issue 1823263002: Add graph intrinsics for many math library functions: (Closed) Base URL: git@github.com:dart-lang/sdk.git@master
Patch Set: Created 4 years, 9 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
« no previous file with comments | « runtime/vm/intrinsifier.h ('k') | runtime/vm/intrinsifier_arm.cc » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
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 // Class for intrinsifying functions. 4 // Class for intrinsifying functions.
5 5
6 #include "vm/assembler.h" 6 #include "vm/assembler.h"
7 #include "vm/compiler.h" 7 #include "vm/compiler.h"
8 #include "vm/flags.h" 8 #include "vm/flags.h"
9 #include "vm/flow_graph.h" 9 #include "vm/flow_graph.h"
10 #include "vm/flow_graph_compiler.h" 10 #include "vm/flow_graph_compiler.h"
11 #include "vm/flow_graph_allocator.h" 11 #include "vm/flow_graph_allocator.h"
12 #include "vm/flow_graph_builder.h" 12 #include "vm/flow_graph_builder.h"
13 #include "vm/il_printer.h" 13 #include "vm/il_printer.h"
14 #include "vm/intermediate_language.h" 14 #include "vm/intermediate_language.h"
15 #include "vm/intrinsifier.h" 15 #include "vm/intrinsifier.h"
16 #include "vm/object.h" 16 #include "vm/object.h"
17 #include "vm/parser.h" 17 #include "vm/parser.h"
18 #include "vm/symbols.h" 18 #include "vm/symbols.h"
19 19
20 20
21 namespace dart { 21 namespace dart {
22 22
23 DEFINE_FLAG(bool, intrinsify, true, "Instrinsify when possible"); 23 DEFINE_FLAG(bool, intrinsify, true, "Instrinsify when possible");
24 DEFINE_FLAG(bool, trace_intrinsifier, false, "Trace intrinsifier");
24 DECLARE_FLAG(bool, code_comments); 25 DECLARE_FLAG(bool, code_comments);
25 DECLARE_FLAG(bool, print_flow_graph); 26 DECLARE_FLAG(bool, print_flow_graph);
26 DECLARE_FLAG(bool, print_flow_graph_optimized); 27 DECLARE_FLAG(bool, print_flow_graph_optimized);
27 28
28 bool Intrinsifier::CanIntrinsify(const Function& function) { 29 bool Intrinsifier::CanIntrinsify(const Function& function) {
30 if (FLAG_trace_intrinsifier) {
31 THR_Print("CanIntrinsify %s ->", function.ToQualifiedCString());
32 }
29 if (!FLAG_intrinsify) return false; 33 if (!FLAG_intrinsify) return false;
30 if (function.IsClosureFunction()) return false; 34 if (function.IsClosureFunction()) {
35 if (FLAG_trace_intrinsifier) {
36 THR_Print("No, closure function.\n");
37 }
38 return false;
39 }
31 // Can occur because of compile-all flag. 40 // Can occur because of compile-all flag.
32 if (function.is_external()) return false; 41 if (function.is_external()) {
33 return function.is_intrinsic(); 42 if (FLAG_trace_intrinsifier) {
43 THR_Print("No, external function.\n");
44 }
45 return false;
46 }
47 if (!function.is_intrinsic()) {
48 if (FLAG_trace_intrinsifier) {
49 THR_Print("No, not intrinsic function.\n");
50 }
51 return false;
52 }
53 if (FLAG_trace_intrinsifier) {
54 THR_Print("Yes.\n");
55 }
56 return true;
34 } 57 }
35 58
36 59
37 #if defined(DART_NO_SNAPSHOT) 60 #if defined(DART_NO_SNAPSHOT)
38 void Intrinsifier::InitializeState() { 61 void Intrinsifier::InitializeState() {
39 Thread* thread = Thread::Current(); 62 Thread* thread = Thread::Current();
40 Zone* zone = thread->zone(); 63 Zone* zone = thread->zone();
41 Library& lib = Library::Handle(zone); 64 Library& lib = Library::Handle(zone);
42 Class& cls = Class::Handle(zone); 65 Class& cls = Class::Handle(zone);
43 Function& func = Function::Handle(zone); 66 Function& func = Function::Handle(zone);
(...skipping 27 matching lines...) Expand all
71 lib = Library::CoreLibrary(); 94 lib = Library::CoreLibrary();
72 ASSERT(!lib.IsNull()); 95 ASSERT(!lib.IsNull());
73 CORE_LIB_INTRINSIC_LIST(SETUP_FUNCTION); 96 CORE_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
74 CORE_INTEGER_LIB_INTRINSIC_LIST(SETUP_FUNCTION); 97 CORE_INTEGER_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
75 GRAPH_CORE_INTRINSICS_LIST(SETUP_FUNCTION); 98 GRAPH_CORE_INTRINSICS_LIST(SETUP_FUNCTION);
76 99
77 // Set up all math lib functions that can be intrinsified. 100 // Set up all math lib functions that can be intrinsified.
78 lib = Library::MathLibrary(); 101 lib = Library::MathLibrary();
79 ASSERT(!lib.IsNull()); 102 ASSERT(!lib.IsNull());
80 MATH_LIB_INTRINSIC_LIST(SETUP_FUNCTION); 103 MATH_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
104 GRAPH_MATH_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
81 105
82 // Set up all dart:typed_data lib functions that can be intrinsified. 106 // Set up all dart:typed_data lib functions that can be intrinsified.
83 lib = Library::TypedDataLibrary(); 107 lib = Library::TypedDataLibrary();
84 ASSERT(!lib.IsNull()); 108 ASSERT(!lib.IsNull());
85 TYPED_DATA_LIB_INTRINSIC_LIST(SETUP_FUNCTION); 109 TYPED_DATA_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
86 GRAPH_TYPED_DATA_INTRINSICS_LIST(SETUP_FUNCTION); 110 GRAPH_TYPED_DATA_INTRINSICS_LIST(SETUP_FUNCTION);
87 111
88 // Setup all dart:developer lib functions that can be intrinsified. 112 // Setup all dart:developer lib functions that can be intrinsified.
89 lib = Library::DeveloperLibrary(); 113 lib = Library::DeveloperLibrary();
90 ASSERT(!lib.IsNull()); 114 ASSERT(!lib.IsNull());
91 DEVELOPER_LIB_INTRINSIC_LIST(SETUP_FUNCTION); 115 DEVELOPER_LIB_INTRINSIC_LIST(SETUP_FUNCTION);
92 116
93 #undef SETUP_FUNCTION 117 #undef SETUP_FUNCTION
94 } 118 }
95 #endif // defined(DART_NO_SNAPSHOT). 119 #endif // defined(DART_NO_SNAPSHOT).
96 120
97
98 static void EmitCodeFor(FlowGraphCompiler* compiler, 121 static void EmitCodeFor(FlowGraphCompiler* compiler,
99 FlowGraph* graph) { 122 FlowGraph* graph) {
100 // The FlowGraph here is constructed by the intrinsics builder methods, and 123 // The FlowGraph here is constructed by the intrinsics builder methods, and
101 // is different from compiler->flow_graph(), the original method's flow graph. 124 // is different from compiler->flow_graph(), the original method's flow graph.
102 compiler->assembler()->Comment("Graph intrinsic"); 125 compiler->assembler()->Comment("Graph intrinsic begin");
103 for (intptr_t i = 0; i < graph->reverse_postorder().length(); i++) { 126 for (intptr_t i = 0; i < graph->reverse_postorder().length(); i++) {
104 BlockEntryInstr* block = graph->reverse_postorder()[i]; 127 BlockEntryInstr* block = graph->reverse_postorder()[i];
105 if (block->IsGraphEntry()) continue; // No code for graph entry needed. 128 if (block->IsGraphEntry()) continue; // No code for graph entry needed.
106 129
107 if (block->HasParallelMove()) { 130 if (block->HasParallelMove()) {
108 compiler->parallel_move_resolver()->EmitNativeCode( 131 compiler->parallel_move_resolver()->EmitNativeCode(
109 block->parallel_move()); 132 block->parallel_move());
110 } 133 }
111 134
112 for (ForwardInstructionIterator it(block); !it.Done(); it.Advance()) { 135 for (ForwardInstructionIterator it(block); !it.Done(); it.Advance()) {
113 Instruction* instr = it.Current(); 136 Instruction* instr = it.Current();
114 if (FLAG_code_comments) compiler->EmitComment(instr); 137 if (FLAG_code_comments) compiler->EmitComment(instr);
115 if (instr->IsParallelMove()) { 138 if (instr->IsParallelMove()) {
116 compiler->parallel_move_resolver()->EmitNativeCode( 139 compiler->parallel_move_resolver()->EmitNativeCode(
117 instr->AsParallelMove()); 140 instr->AsParallelMove());
141 } else if (instr->IsInvokeMathCFunction()) {
142 ASSERT(instr->locs() != NULL);
143 Intrinsifier::IntrinsicCallPrologue(compiler->assembler());
144 instr->EmitNativeCode(compiler);
145 Intrinsifier::IntrinsicCallEpilogue(compiler->assembler());
118 } else { 146 } else {
119 ASSERT(instr->locs() != NULL); 147 ASSERT(instr->locs() != NULL);
120 // Calls are not supported in intrinsics code. 148 // Calls are not supported in intrinsics code.
121 ASSERT(!instr->locs()->always_calls()); 149 ASSERT(!instr->locs()->always_calls());
122 instr->EmitNativeCode(compiler); 150 instr->EmitNativeCode(compiler);
123 } 151 }
124 } 152 }
125 } 153 }
154 compiler->assembler()->Comment("Graph intrinsic end");
126 } 155 }
127 156
128 157
129 bool Intrinsifier::GraphIntrinsify(const ParsedFunction& parsed_function, 158 bool Intrinsifier::GraphIntrinsify(const ParsedFunction& parsed_function,
130 FlowGraphCompiler* compiler) { 159 FlowGraphCompiler* compiler) {
131 ZoneGrowableArray<const ICData*>* ic_data_array = 160 ZoneGrowableArray<const ICData*>* ic_data_array =
132 new ZoneGrowableArray<const ICData*>(); 161 new ZoneGrowableArray<const ICData*>();
133 FlowGraphBuilder builder(parsed_function, 162 FlowGraphBuilder builder(parsed_function,
134 *ic_data_array, 163 *ic_data_array,
135 NULL, // NULL = not inlining. 164 NULL, // NULL = not inlining.
(...skipping 81 matching lines...) Expand 10 before | Expand all | Expand 10 after
217 return kFloat32x4Cid; 246 return kFloat32x4Cid;
218 case kUnboxedUint32: 247 case kUnboxedUint32:
219 return kDynamicCid; // smi or mint. 248 return kDynamicCid; // smi or mint.
220 default: 249 default:
221 UNREACHABLE(); 250 UNREACHABLE();
222 return kIllegalCid; 251 return kIllegalCid;
223 } 252 }
224 } 253 }
225 254
226 255
256 // Notes about the graph intrinsics:
257 //
258 // IR instructions which would jump to a deoptimization sequence on failure
259 // instead branch to the intrinsic slow path.
260 //
227 class BlockBuilder : public ValueObject { 261 class BlockBuilder : public ValueObject {
228 public: 262 public:
229 BlockBuilder(FlowGraph* flow_graph, TargetEntryInstr* entry) 263 BlockBuilder(FlowGraph* flow_graph, TargetEntryInstr* entry)
230 : flow_graph_(flow_graph), entry_(entry), current_(entry) { } 264 : flow_graph_(flow_graph), entry_(entry), current_(entry) { }
231 265
232 Definition* AddToInitialDefinitions(Definition* def) { 266 Definition* AddToInitialDefinitions(Definition* def) {
233 def->set_ssa_temp_index(flow_graph_->alloc_ssa_temp_index()); 267 def->set_ssa_temp_index(flow_graph_->alloc_ssa_temp_index());
234 flow_graph_->AddToInitialDefinitions(def); 268 flow_graph_->AddToInitialDefinitions(def);
235 return def; 269 return def;
236 } 270 }
(...skipping 25 matching lines...) Expand all
262 296
263 TokenPosition TokenPos() { 297 TokenPosition TokenPos() {
264 return flow_graph_->function().token_pos(); 298 return flow_graph_->function().token_pos();
265 } 299 }
266 300
267 Definition* AddNullDefinition() { 301 Definition* AddNullDefinition() {
268 return AddDefinition( 302 return AddDefinition(
269 new ConstantInstr(Object::ZoneHandle(Object::null()))); 303 new ConstantInstr(Object::ZoneHandle(Object::null())));
270 } 304 }
271 305
272 Definition* AddUnboxInstr(Representation rep, Value* value) { 306 Definition* AddUnboxInstr(Representation rep,
307 Value* value,
308 bool is_checked) {
273 Definition* unboxed_value = AddDefinition( 309 Definition* unboxed_value = AddDefinition(
274 UnboxInstr::Create(rep, value, Thread::kNoDeoptId)); 310 UnboxInstr::Create(rep, value, Thread::kNoDeoptId));
275 // Manually adjust reaching type because there is no type propagation 311 if (is_checked) {
276 // when building intrinsics. 312 // The type of |value| has already been checked and it is safe to
277 unboxed_value->AsUnbox()->value()->SetReachingType(ZoneCompileType::Wrap( 313 // adjust reaching type. This is done manually because there is no type
278 CompileType::FromCid(CidForRepresentation(rep)))); 314 // propagation when building intrinsics.
315 unboxed_value->AsUnbox()->value()->SetReachingType(ZoneCompileType::Wrap(
316 CompileType::FromCid(CidForRepresentation(rep))));
317 }
279 return unboxed_value; 318 return unboxed_value;
280 } 319 }
281 320
321 Definition* AddUnboxInstr(Representation rep,
322 Definition* boxed,
323 bool is_checked) {
324 return AddUnboxInstr(rep, new Value(boxed), is_checked);
325 }
326
327 Definition* InvokeMathCFunction(MethodRecognizer::Kind recognized_kind,
328 ZoneGrowableArray<Value*>* args) {
329 return InvokeMathCFunctionHelper(recognized_kind, args);
330 }
331
282 private: 332 private:
333 Definition* InvokeMathCFunctionHelper(MethodRecognizer::Kind recognized_kind,
334 ZoneGrowableArray<Value*>* args) {
335 InvokeMathCFunctionInstr* invoke_math_c_function =
336 new InvokeMathCFunctionInstr(args,
337 Thread::kNoDeoptId,
338 recognized_kind,
339 TokenPos());
340 AddDefinition(invoke_math_c_function);
341 return invoke_math_c_function;
342 }
343
344
283 FlowGraph* flow_graph_; 345 FlowGraph* flow_graph_;
284 BlockEntryInstr* entry_; 346 BlockEntryInstr* entry_;
285 Instruction* current_; 347 Instruction* current_;
286 }; 348 };
287 349
288 350
289 static void PrepareIndexedOp(BlockBuilder* builder, 351 static void PrepareIndexedOp(BlockBuilder* builder,
290 Definition* array, 352 Definition* array,
291 Definition* index, 353 Definition* index,
292 intptr_t length_offset) { 354 intptr_t length_offset) {
(...skipping 160 matching lines...) Expand 10 before | Expand all | Expand 10 after
453 TargetEntryInstr* normal_entry = graph_entry->normal_entry(); 515 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
454 BlockBuilder builder(flow_graph, normal_entry); 516 BlockBuilder builder(flow_graph, normal_entry);
455 517
456 Definition* value = builder.AddParameter(1); 518 Definition* value = builder.AddParameter(1);
457 Definition* index = builder.AddParameter(2); 519 Definition* index = builder.AddParameter(2);
458 Definition* array = builder.AddParameter(3); 520 Definition* array = builder.AddParameter(3);
459 521
460 PrepareIndexedOp(&builder, array, index, TypedData::length_offset()); 522 PrepareIndexedOp(&builder, array, index, TypedData::length_offset());
461 523
462 Definition* unboxed_value = 524 Definition* unboxed_value =
463 builder.AddUnboxInstr(kUnboxedUint32, new Value(value)); 525 builder.AddUnboxInstr(kUnboxedUint32,
526 new Value(value),
527 /* is_checked = */ true);
464 528
465 builder.AddInstruction( 529 builder.AddInstruction(
466 new StoreIndexedInstr(new Value(array), 530 new StoreIndexedInstr(new Value(array),
467 new Value(index), 531 new Value(index),
468 new Value(unboxed_value), 532 new Value(unboxed_value),
469 kNoStoreBarrier, 533 kNoStoreBarrier,
470 4, // index scale 534 4, // index scale
471 kTypedDataUint32ArrayCid, 535 kTypedDataUint32ArrayCid,
472 Thread::kNoDeoptId, 536 Thread::kNoDeoptId,
473 builder.TokenPos())); 537 builder.TokenPos()));
(...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after
521 Object::empty_array(), // Dummy args. descr. 585 Object::empty_array(), // Dummy args. descr.
522 Thread::kNoDeoptId, 586 Thread::kNoDeoptId,
523 1)); 587 1));
524 value_check.AddReceiverCheck(kDoubleCid, flow_graph->function()); 588 value_check.AddReceiverCheck(kDoubleCid, flow_graph->function());
525 builder.AddInstruction( 589 builder.AddInstruction(
526 new CheckClassInstr(new Value(value), 590 new CheckClassInstr(new Value(value),
527 Thread::kNoDeoptId, 591 Thread::kNoDeoptId,
528 value_check, 592 value_check,
529 builder.TokenPos())); 593 builder.TokenPos()));
530 Definition* double_value = 594 Definition* double_value =
531 builder.AddUnboxInstr(kUnboxedDouble, new Value(value)); 595 builder.AddUnboxInstr(kUnboxedDouble,
596 new Value(value),
597 /* is_checked = */ true);
532 598
533 builder.AddInstruction( 599 builder.AddInstruction(
534 new StoreIndexedInstr(new Value(array), 600 new StoreIndexedInstr(new Value(array),
535 new Value(index), 601 new Value(index),
536 new Value(double_value), 602 new Value(double_value),
537 kNoStoreBarrier, 603 kNoStoreBarrier,
538 8, // index scale 604 8, // index scale
539 kTypedDataFloat64ArrayCid, 605 kTypedDataFloat64ArrayCid,
540 Thread::kNoDeoptId, 606 Thread::kNoDeoptId,
541 builder.TokenPos())); 607 builder.TokenPos()));
(...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after
589 Thread::kNoDeoptId, 655 Thread::kNoDeoptId,
590 1)); 656 1));
591 value_check.AddReceiverCheck(kFloat32x4Cid, flow_graph->function()); 657 value_check.AddReceiverCheck(kFloat32x4Cid, flow_graph->function());
592 // Check argument. Receiver (left) is known to be a Float32x4. 658 // Check argument. Receiver (left) is known to be a Float32x4.
593 builder.AddInstruction( 659 builder.AddInstruction(
594 new CheckClassInstr(new Value(right), 660 new CheckClassInstr(new Value(right),
595 Thread::kNoDeoptId, 661 Thread::kNoDeoptId,
596 value_check, 662 value_check,
597 builder.TokenPos())); 663 builder.TokenPos()));
598 Definition* left_simd = 664 Definition* left_simd =
599 builder.AddUnboxInstr(kUnboxedFloat32x4, new Value(left)); 665 builder.AddUnboxInstr(kUnboxedFloat32x4,
666 new Value(left),
667 /* is_checked = */ true);
600 668
601 Definition* right_simd = 669 Definition* right_simd =
602 builder.AddUnboxInstr(kUnboxedFloat32x4, new Value(right)); 670 builder.AddUnboxInstr(kUnboxedFloat32x4,
671 new Value(right),
672 /* is_checked = */ true);
603 673
604 Definition* unboxed_result = builder.AddDefinition( 674 Definition* unboxed_result = builder.AddDefinition(
605 new BinaryFloat32x4OpInstr(kind, 675 new BinaryFloat32x4OpInstr(kind,
606 new Value(left_simd), 676 new Value(left_simd),
607 new Value(right_simd), 677 new Value(right_simd),
608 Thread::kNoDeoptId)); 678 Thread::kNoDeoptId));
609 Definition* result = builder.AddDefinition( 679 Definition* result = builder.AddDefinition(
610 BoxInstr::Create(kUnboxedFloat32x4, new Value(unboxed_result))); 680 BoxInstr::Create(kUnboxedFloat32x4, new Value(unboxed_result)));
611 builder.AddIntrinsicReturn(new Value(result)); 681 builder.AddIntrinsicReturn(new Value(result));
612 return true; 682 return true;
(...skipping 18 matching lines...) Expand all
631 static bool BuildFloat32x4Shuffle(FlowGraph* flow_graph, 701 static bool BuildFloat32x4Shuffle(FlowGraph* flow_graph,
632 MethodRecognizer::Kind kind) { 702 MethodRecognizer::Kind kind) {
633 if (!FlowGraphCompiler::SupportsUnboxedSimd128()) return false; 703 if (!FlowGraphCompiler::SupportsUnboxedSimd128()) return false;
634 GraphEntryInstr* graph_entry = flow_graph->graph_entry(); 704 GraphEntryInstr* graph_entry = flow_graph->graph_entry();
635 TargetEntryInstr* normal_entry = graph_entry->normal_entry(); 705 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
636 BlockBuilder builder(flow_graph, normal_entry); 706 BlockBuilder builder(flow_graph, normal_entry);
637 707
638 Definition* receiver = builder.AddParameter(1); 708 Definition* receiver = builder.AddParameter(1);
639 709
640 Definition* unboxed_receiver = 710 Definition* unboxed_receiver =
641 builder.AddUnboxInstr(kUnboxedFloat32x4, new Value(receiver)); 711 builder.AddUnboxInstr(kUnboxedFloat32x4,
712 new Value(receiver),
713 /* is_checked = */ true);
642 714
643 Definition* unboxed_result = builder.AddDefinition( 715 Definition* unboxed_result = builder.AddDefinition(
644 new Simd32x4ShuffleInstr(kind, 716 new Simd32x4ShuffleInstr(kind,
645 new Value(unboxed_receiver), 717 new Value(unboxed_receiver),
646 0, 718 0,
647 Thread::kNoDeoptId)); 719 Thread::kNoDeoptId));
648 720
649 Definition* result = builder.AddDefinition( 721 Definition* result = builder.AddDefinition(
650 BoxInstr::Create(kUnboxedDouble, new Value(unboxed_result))); 722 BoxInstr::Create(kUnboxedDouble, new Value(unboxed_result)));
651 builder.AddIntrinsicReturn(new Value(result)); 723 builder.AddIntrinsicReturn(new Value(result));
(...skipping 213 matching lines...) Expand 10 before | Expand all | Expand 10 after
865 } 937 }
866 938
867 939
868 bool Intrinsifier::Build_DoubleFlipSignBit(FlowGraph* flow_graph) { 940 bool Intrinsifier::Build_DoubleFlipSignBit(FlowGraph* flow_graph) {
869 GraphEntryInstr* graph_entry = flow_graph->graph_entry(); 941 GraphEntryInstr* graph_entry = flow_graph->graph_entry();
870 TargetEntryInstr* normal_entry = graph_entry->normal_entry(); 942 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
871 BlockBuilder builder(flow_graph, normal_entry); 943 BlockBuilder builder(flow_graph, normal_entry);
872 944
873 Definition* receiver = builder.AddParameter(1); 945 Definition* receiver = builder.AddParameter(1);
874 Definition* unboxed_value = 946 Definition* unboxed_value =
875 builder.AddUnboxInstr(kUnboxedDouble, new Value(receiver)); 947 builder.AddUnboxInstr(kUnboxedDouble,
948 new Value(receiver),
949 /* is_checked = */ true);
876 Definition* unboxed_result = builder.AddDefinition( 950 Definition* unboxed_result = builder.AddDefinition(
877 new UnaryDoubleOpInstr(Token::kNEGATE, 951 new UnaryDoubleOpInstr(Token::kNEGATE,
878 new Value(unboxed_value), 952 new Value(unboxed_value),
879 Thread::kNoDeoptId)); 953 Thread::kNoDeoptId));
880 Definition* result = builder.AddDefinition( 954 Definition* result = builder.AddDefinition(
881 BoxInstr::Create(kUnboxedDouble, new Value(unboxed_result))); 955 BoxInstr::Create(kUnboxedDouble, new Value(unboxed_result)));
882 builder.AddIntrinsicReturn(new Value(result)); 956 builder.AddIntrinsicReturn(new Value(result));
883 return true; 957 return true;
884 } 958 }
885 959
960
961 static bool BuildInvokeMathCFunction(BlockBuilder* builder,
962 MethodRecognizer::Kind kind,
963 intptr_t num_parameters = 1) {
964 ZoneGrowableArray<Value*>* args =
965 new ZoneGrowableArray<Value*>(num_parameters);
966
967 for (intptr_t i = 0; i < num_parameters; i++) {
968 const intptr_t parameter_index = (num_parameters - i);
969 Definition* value = builder->AddParameter(parameter_index);
970 Definition* unboxed_value =
971 builder->AddUnboxInstr(kUnboxedDouble, value, /* is_checked = */ false);
972 args->Add(new Value(unboxed_value));
973 }
974
975 Definition* unboxed_result =
976 builder->InvokeMathCFunction(kind, args);
977
978 Definition* result = builder->AddDefinition(
979 BoxInstr::Create(kUnboxedDouble, new Value(unboxed_result)));
980
981 builder->AddIntrinsicReturn(new Value(result));
982
983 return true;
984 }
985
986
987 bool Intrinsifier::Build_MathSin(FlowGraph* flow_graph) {
988 if (!FlowGraphCompiler::SupportsUnboxedDoubles()) return false;
989
990 GraphEntryInstr* graph_entry = flow_graph->graph_entry();
991 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
992 BlockBuilder builder(flow_graph, normal_entry);
993
994 return BuildInvokeMathCFunction(&builder,
995 MethodRecognizer::kMathSin);
996 }
997
998
999 bool Intrinsifier::Build_MathCos(FlowGraph* flow_graph) {
1000 if (!FlowGraphCompiler::SupportsUnboxedDoubles()) return false;
1001
1002 GraphEntryInstr* graph_entry = flow_graph->graph_entry();
1003 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
1004 BlockBuilder builder(flow_graph, normal_entry);
1005
1006 return BuildInvokeMathCFunction(&builder,
1007 MethodRecognizer::kMathCos);
1008 }
1009
1010
1011 bool Intrinsifier::Build_MathTan(FlowGraph* flow_graph) {
1012 if (!FlowGraphCompiler::SupportsUnboxedDoubles()) return false;
1013
1014 GraphEntryInstr* graph_entry = flow_graph->graph_entry();
1015 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
1016 BlockBuilder builder(flow_graph, normal_entry);
1017
1018 return BuildInvokeMathCFunction(&builder,
1019 MethodRecognizer::kMathTan);
1020 }
1021
1022
1023 bool Intrinsifier::Build_MathAsin(FlowGraph* flow_graph) {
1024 if (!FlowGraphCompiler::SupportsUnboxedDoubles()) return false;
1025
1026 GraphEntryInstr* graph_entry = flow_graph->graph_entry();
1027 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
1028 BlockBuilder builder(flow_graph, normal_entry);
1029
1030 return BuildInvokeMathCFunction(&builder,
1031 MethodRecognizer::kMathAsin);
1032 }
1033
1034
1035 bool Intrinsifier::Build_MathAcos(FlowGraph* flow_graph) {
1036 if (!FlowGraphCompiler::SupportsUnboxedDoubles()) return false;
1037
1038 GraphEntryInstr* graph_entry = flow_graph->graph_entry();
1039 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
1040 BlockBuilder builder(flow_graph, normal_entry);
1041
1042 return BuildInvokeMathCFunction(&builder,
1043 MethodRecognizer::kMathAcos);
1044 }
1045
1046
1047 bool Intrinsifier::Build_MathAtan(FlowGraph* flow_graph) {
1048 if (!FlowGraphCompiler::SupportsUnboxedDoubles()) return false;
1049
1050 GraphEntryInstr* graph_entry = flow_graph->graph_entry();
1051 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
1052 BlockBuilder builder(flow_graph, normal_entry);
1053
1054 return BuildInvokeMathCFunction(&builder,
1055 MethodRecognizer::kMathAtan);
1056 }
1057
1058
1059 bool Intrinsifier::Build_MathAtan2(FlowGraph* flow_graph) {
1060 if (!FlowGraphCompiler::SupportsUnboxedDoubles()) return false;
1061
1062 GraphEntryInstr* graph_entry = flow_graph->graph_entry();
1063 TargetEntryInstr* normal_entry = graph_entry->normal_entry();
1064 BlockBuilder builder(flow_graph, normal_entry);
1065
1066 return BuildInvokeMathCFunction(&builder,
1067 MethodRecognizer::kMathAtan2,
1068 /* num_parameters = */ 2);
1069 }
1070
886 } // namespace dart 1071 } // namespace dart
OLDNEW
« no previous file with comments | « runtime/vm/intrinsifier.h ('k') | runtime/vm/intrinsifier_arm.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698