Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright 2015 the V8 project authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "src/v8.h" | |
| 6 | |
| 7 #include "src/compiler/pipeline.h" | |
| 8 #include "src/execution.h" | |
| 9 #include "src/handles.h" | |
| 10 #include "src/interpreter/bytecode-array-builder.h" | |
| 11 #include "src/interpreter/interpreter.h" | |
| 12 #include "src/parser.h" | |
| 13 #include "test/cctest/cctest.h" | |
| 14 | |
| 15 namespace v8 { | |
| 16 namespace internal { | |
| 17 namespace compiler { | |
| 18 | |
| 19 | |
| 20 static const char* kFunctionName = "f"; | |
| 21 | |
| 22 | |
| 23 static MaybeHandle<Object> CallFunction(Isolate* isolate, | |
| 24 Handle<JSFunction> function) { | |
| 25 return Execution::Call(isolate, function, | |
| 26 isolate->factory()->undefined_value(), 0, nullptr, | |
| 27 false); | |
| 28 } | |
| 29 | |
| 30 | |
| 31 template <class... A> | |
| 32 static MaybeHandle<Object> CallFunction(Isolate* isolate, | |
| 33 Handle<JSFunction> function, | |
| 34 A... args) { | |
| 35 Handle<Object> argv[] = {args...}; | |
| 36 return Execution::Call(isolate, function, | |
| 37 isolate->factory()->undefined_value(), sizeof...(args), | |
|
Michael Starzinger
2015/09/03 12:19:40
Wow, "sizeof...", C++ never seizes to surprise me.
| |
| 38 argv, false); | |
| 39 } | |
| 40 | |
| 41 | |
| 42 template <class... A> | |
| 43 class BytecodeGraphCallable { | |
| 44 public: | |
| 45 BytecodeGraphCallable(Isolate* isolate, Handle<JSFunction> function) | |
| 46 : isolate_(isolate), function_(function) {} | |
| 47 virtual ~BytecodeGraphCallable() {} | |
| 48 | |
| 49 MaybeHandle<Object> operator()(A... args) { | |
| 50 return CallFunction(isolate_, function_, args...); | |
| 51 } | |
| 52 | |
| 53 private: | |
| 54 Isolate* isolate_; | |
| 55 Handle<JSFunction> function_; | |
| 56 }; | |
| 57 | |
| 58 | |
| 59 class BytecodeGraphTester { | |
| 60 public: | |
| 61 BytecodeGraphTester(Isolate* isolate, Zone* zone, const char* script) | |
| 62 : isolate_(isolate), zone_(zone), script_(script) { | |
| 63 i::FLAG_ignition = true; | |
| 64 i::FLAG_ignition_filter = kFunctionName; | |
|
Michael Starzinger
2015/09/03 12:19:40
This assignment could potentially trigger double-f
oth
2015/09/04 11:06:36
Thanks, we had another instance of this in the int
| |
| 65 i::FLAG_always_opt = false; | |
| 66 // Ensure handler table is generated. | |
| 67 isolate->interpreter()->Initialize(); | |
| 68 } | |
| 69 virtual ~BytecodeGraphTester() {} | |
| 70 | |
| 71 template <class... A> | |
| 72 BytecodeGraphCallable<A...> GetCallable() { | |
| 73 return BytecodeGraphCallable<A...>(isolate_, GetFunction()); | |
| 74 } | |
| 75 | |
| 76 private: | |
| 77 Isolate* isolate_; | |
| 78 Zone* zone_; | |
| 79 const char* script_; | |
| 80 | |
| 81 Handle<JSFunction> GetFunction() { | |
| 82 CompileRun(script_); | |
| 83 Local<Function> api_function = | |
| 84 Local<Function>::Cast(CcTest::global()->Get(v8_str(kFunctionName))); | |
| 85 Handle<JSFunction> function = v8::Utils::OpenHandle(*api_function); | |
| 86 CHECK(function->shared()->HasBytecodeArray()); | |
| 87 | |
| 88 ParseInfo parse_info(zone_, function); | |
| 89 | |
| 90 CompilationInfo compilation_info(&parse_info); | |
| 91 compilation_info.SetOptimizing(BailoutId::None(), Handle<Code>()); | |
| 92 Parser parser(&parse_info); | |
| 93 CHECK(parser.Parse(&parse_info)); | |
| 94 compiler::Pipeline pipeline(&compilation_info); | |
| 95 Handle<Code> code = pipeline.GenerateCode(); | |
| 96 function->ReplaceCode(*code); | |
| 97 | |
| 98 return function; | |
| 99 } | |
| 100 | |
| 101 DISALLOW_COPY_AND_ASSIGN(BytecodeGraphTester); | |
| 102 }; | |
| 103 | |
| 104 } // namespace compiler | |
| 105 } // namespace internal | |
| 106 } // namespace v8 | |
| 107 | |
| 108 | |
| 109 using namespace v8::internal; | |
| 110 using namespace v8::internal::compiler; | |
| 111 | |
| 112 | |
| 113 struct ExpectedSnippet { | |
| 114 const char* code_snippet; | |
| 115 Handle<Object> return_val; | |
| 116 }; | |
| 117 | |
| 118 | |
| 119 TEST(BytecodeGraphBuilderReturnStatements) { | |
| 120 HandleAndZoneScope handles; | |
| 121 Isolate* isolate = handles.main_isolate(); | |
| 122 Factory* factory = isolate->factory(); | |
| 123 | |
| 124 ExpectedSnippet snippets[] = { | |
| 125 {"return;", factory->undefined_value()}, | |
| 126 {"return null;", factory->null_value()}, | |
| 127 {"return true;", factory->true_value()}, | |
| 128 {"return false;", factory->false_value()}, | |
| 129 {"return 0;", Handle<Object>(Smi::FromInt(0), isolate)}, | |
| 130 {"return +1;", Handle<Object>(Smi::FromInt(1), isolate)}, | |
| 131 {"return -1;", Handle<Object>(Smi::FromInt(-1), isolate)}, | |
| 132 {"return +127;", Handle<Object>(Smi::FromInt(127), isolate)}, | |
| 133 {"return -128;", Handle<Object>(Smi::FromInt(-128), isolate)}, | |
| 134 }; | |
| 135 | |
| 136 size_t num_snippets = sizeof(snippets) / sizeof(snippets[0]); | |
| 137 for (size_t i = 0; i < num_snippets; i++) { | |
| 138 ScopedVector<char> script(1024); | |
| 139 SNPrintF(script, "function %s() { %s }\n%s();", kFunctionName, | |
| 140 snippets[i].code_snippet, kFunctionName); | |
| 141 | |
| 142 BytecodeGraphTester tester(handles.main_isolate(), handles.main_zone(), | |
| 143 script.start()); | |
| 144 auto callable = tester.GetCallable<>(); | |
| 145 Handle<Object> return_val = callable().ToHandleChecked(); | |
| 146 CHECK(return_val.is_identical_to(snippets[i].return_val)); | |
| 147 } | |
| 148 } | |
| 149 | |
| 150 | |
| 151 TEST(BytecodeGraphBuilderPrimitiveExpressions) { | |
| 152 HandleAndZoneScope handles; | |
| 153 Isolate* isolate = handles.main_isolate(); | |
| 154 | |
| 155 ExpectedSnippet snippets[] = { | |
| 156 {"return 1 + 1;", Handle<Object>(Smi::FromInt(2), isolate)}, | |
| 157 {"return 20 - 30;", Handle<Object>(Smi::FromInt(-10), isolate)}, | |
| 158 {"return 4 * 100;", Handle<Object>(Smi::FromInt(400), isolate)}, | |
| 159 {"return 100 / 5;", Handle<Object>(Smi::FromInt(20), isolate)}, | |
| 160 {"return 25 % 7;", Handle<Object>(Smi::FromInt(4), isolate)}, | |
| 161 }; | |
| 162 | |
| 163 size_t num_snippets = sizeof(snippets) / sizeof(snippets[0]); | |
| 164 for (size_t i = 0; i < num_snippets; i++) { | |
| 165 ScopedVector<char> script(1024); | |
| 166 SNPrintF(script, "function %s() { %s }\n%s();", kFunctionName, | |
| 167 snippets[i].code_snippet, kFunctionName); | |
| 168 | |
| 169 BytecodeGraphTester tester(handles.main_isolate(), handles.main_zone(), | |
| 170 script.start()); | |
| 171 auto callable = tester.GetCallable<>(); | |
| 172 Handle<Object> return_val = callable().ToHandleChecked(); | |
| 173 CHECK(return_val.is_identical_to(snippets[i].return_val)); | |
| 174 } | |
| 175 } | |
| OLD | NEW |