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 <stdlib.h> |
| 6 |
| 7 #include "src/v8.h" |
| 8 |
| 9 #include "src/ast.h" |
| 10 #include "src/ast-expression-visitor.h" |
| 11 #include "src/parser.h" |
| 12 #include "src/rewriter.h" |
| 13 #include "src/scopes.h" |
| 14 #include "src/typing-reset.h" |
| 15 #include "test/cctest/cctest.h" |
| 16 #include "test/cctest/compiler/function-tester.h" |
| 17 #include "test/cctest/expression-type-collector.h" |
| 18 |
| 19 using namespace v8::internal; |
| 20 |
| 21 namespace { |
| 22 |
| 23 class TypeSetter : public AstExpressionVisitor { |
| 24 public: |
| 25 explicit TypeSetter(CompilationInfo* info) : AstExpressionVisitor(info) {} |
| 26 |
| 27 protected: |
| 28 void VisitExpression(Expression* expression) { |
| 29 expression->set_bounds(Bounds(Type::Integral32())); |
| 30 } |
| 31 }; |
| 32 |
| 33 |
| 34 void CheckAllSame(ZoneVector<ExpressionTypeEntry>& types, |
| 35 Bounds expected_type) { |
| 36 HandleAndZoneScope handles; |
| 37 CHECK_TYPES_BEGIN { |
| 38 // function logSum |
| 39 CHECK_EXPR(FunctionLiteral, expected_type) { |
| 40 CHECK_EXPR(FunctionLiteral, expected_type) { |
| 41 CHECK_EXPR(Assignment, expected_type) { |
| 42 CHECK_VAR(start, expected_type); |
| 43 CHECK_EXPR(BinaryOperation, expected_type) { |
| 44 CHECK_VAR(start, expected_type); |
| 45 CHECK_EXPR(Literal, expected_type); |
| 46 } |
| 47 } |
| 48 CHECK_EXPR(Assignment, expected_type) { |
| 49 CHECK_VAR(end, expected_type); |
| 50 CHECK_EXPR(BinaryOperation, expected_type) { |
| 51 CHECK_VAR(end, expected_type); |
| 52 CHECK_EXPR(Literal, expected_type); |
| 53 } |
| 54 } |
| 55 CHECK_EXPR(Assignment, expected_type) { |
| 56 CHECK_VAR(sum, expected_type); |
| 57 CHECK_EXPR(Literal, expected_type); |
| 58 } |
| 59 CHECK_EXPR(Assignment, expected_type) { |
| 60 CHECK_VAR(p, expected_type); |
| 61 CHECK_EXPR(Literal, expected_type); |
| 62 } |
| 63 CHECK_EXPR(Assignment, expected_type) { |
| 64 CHECK_VAR(q, expected_type); |
| 65 CHECK_EXPR(Literal, expected_type); |
| 66 } |
| 67 // for (p = start << 3, q = end << 3; |
| 68 CHECK_EXPR(BinaryOperation, expected_type) { |
| 69 CHECK_EXPR(Assignment, expected_type) { |
| 70 CHECK_VAR(p, expected_type); |
| 71 CHECK_EXPR(BinaryOperation, expected_type) { |
| 72 CHECK_VAR(start, expected_type); |
| 73 CHECK_EXPR(Literal, expected_type); |
| 74 } |
| 75 } |
| 76 CHECK_EXPR(Assignment, expected_type) { |
| 77 CHECK_VAR(q, expected_type); |
| 78 CHECK_EXPR(BinaryOperation, expected_type) { |
| 79 CHECK_VAR(end, expected_type); |
| 80 CHECK_EXPR(Literal, expected_type); |
| 81 } |
| 82 } |
| 83 } |
| 84 // (p|0) < (q|0); |
| 85 CHECK_EXPR(CompareOperation, expected_type) { |
| 86 CHECK_EXPR(BinaryOperation, expected_type) { |
| 87 CHECK_VAR(p, expected_type); |
| 88 CHECK_EXPR(Literal, expected_type); |
| 89 } |
| 90 CHECK_EXPR(BinaryOperation, expected_type) { |
| 91 CHECK_VAR(q, expected_type); |
| 92 CHECK_EXPR(Literal, expected_type); |
| 93 } |
| 94 } |
| 95 // p = (p + 8)|0) {\n" |
| 96 CHECK_EXPR(Assignment, expected_type) { |
| 97 CHECK_VAR(p, expected_type); |
| 98 CHECK_EXPR(BinaryOperation, expected_type) { |
| 99 CHECK_EXPR(BinaryOperation, expected_type) { |
| 100 CHECK_VAR(p, expected_type); |
| 101 CHECK_EXPR(Literal, expected_type); |
| 102 } |
| 103 CHECK_EXPR(Literal, expected_type); |
| 104 } |
| 105 } |
| 106 // sum = sum + +log(values[p>>3]); |
| 107 CHECK_EXPR(Assignment, expected_type) { |
| 108 CHECK_VAR(sum, expected_type); |
| 109 CHECK_EXPR(BinaryOperation, expected_type) { |
| 110 CHECK_VAR(sum, expected_type); |
| 111 CHECK_EXPR(BinaryOperation, expected_type) { |
| 112 CHECK_EXPR(Call, expected_type) { |
| 113 CHECK_VAR(log, expected_type); |
| 114 CHECK_VAR(values, expected_type); |
| 115 CHECK_EXPR(BinaryOperation, expected_type) { |
| 116 CHECK_VAR(p, expected_type); |
| 117 CHECK_EXPR(Literal, expected_type); |
| 118 } |
| 119 } |
| 120 CHECK_EXPR(Literal, expected_type); |
| 121 } |
| 122 } |
| 123 } |
| 124 // return +sum; |
| 125 CHECK_EXPR(BinaryOperation, expected_type) { |
| 126 CHECK_VAR(sum, expected_type); |
| 127 CHECK_EXPR(Literal, expected_type); |
| 128 } |
| 129 } |
| 130 // function geometricMean |
| 131 CHECK_EXPR(FunctionLiteral, expected_type) { |
| 132 CHECK_EXPR(Assignment, expected_type) { |
| 133 CHECK_VAR(start, expected_type); |
| 134 CHECK_EXPR(BinaryOperation, expected_type) { |
| 135 CHECK_VAR(start, expected_type); |
| 136 CHECK_EXPR(Literal, expected_type); |
| 137 } |
| 138 } |
| 139 CHECK_EXPR(Assignment, expected_type) { |
| 140 CHECK_VAR(end, expected_type); |
| 141 CHECK_EXPR(BinaryOperation, expected_type) { |
| 142 CHECK_VAR(end, expected_type); |
| 143 CHECK_EXPR(Literal, expected_type); |
| 144 } |
| 145 } |
| 146 // return +exp(+logSum(start, end) / +((end - start)|0)); |
| 147 CHECK_EXPR(BinaryOperation, expected_type) { |
| 148 CHECK_EXPR(Call, expected_type) { |
| 149 CHECK_VAR(exp, expected_type); |
| 150 CHECK_EXPR(BinaryOperation, expected_type) { |
| 151 CHECK_EXPR(BinaryOperation, expected_type) { |
| 152 CHECK_EXPR(Call, expected_type) { |
| 153 CHECK_VAR(logSum, expected_type); |
| 154 CHECK_VAR(start, expected_type); |
| 155 CHECK_VAR(end, expected_type); |
| 156 } |
| 157 CHECK_EXPR(Literal, expected_type); |
| 158 } |
| 159 CHECK_EXPR(BinaryOperation, expected_type) { |
| 160 CHECK_EXPR(BinaryOperation, expected_type) { |
| 161 CHECK_EXPR(BinaryOperation, expected_type) { |
| 162 CHECK_VAR(end, expected_type); |
| 163 CHECK_VAR(start, expected_type); |
| 164 } |
| 165 CHECK_EXPR(Literal, expected_type); |
| 166 } |
| 167 CHECK_EXPR(Literal, expected_type); |
| 168 } |
| 169 } |
| 170 } |
| 171 CHECK_EXPR(Literal, expected_type); |
| 172 } |
| 173 } |
| 174 // "use asm"; |
| 175 CHECK_EXPR(Literal, expected_type); |
| 176 // var exp = stdlib.Math.exp; |
| 177 CHECK_EXPR(Assignment, expected_type) { |
| 178 CHECK_VAR(exp, expected_type); |
| 179 CHECK_VAR(stdlib, expected_type); |
| 180 CHECK_EXPR(Literal, expected_type); |
| 181 CHECK_EXPR(Literal, expected_type); |
| 182 } |
| 183 // var log = stdlib.Math.log; |
| 184 CHECK_EXPR(Assignment, expected_type) { |
| 185 CHECK_VAR(log, expected_type); |
| 186 CHECK_VAR(stdlib, expected_type); |
| 187 CHECK_EXPR(Literal, expected_type); |
| 188 CHECK_EXPR(Literal, expected_type); |
| 189 } |
| 190 // var values = new stdlib.Float64Array(buffer); |
| 191 CHECK_EXPR(Assignment, expected_type) { |
| 192 CHECK_VAR(values, expected_type); |
| 193 CHECK_EXPR(CallNew, expected_type) { |
| 194 CHECK_VAR(stdlib, expected_type); |
| 195 CHECK_EXPR(Literal, expected_type); |
| 196 CHECK_VAR(buffer, expected_type); |
| 197 } |
| 198 } |
| 199 // return { geometricMean: geometricMean }; |
| 200 CHECK_EXPR(ObjectLiteral, expected_type) { |
| 201 CHECK_VAR(geometricMean, expected_type); |
| 202 } |
| 203 } |
| 204 } |
| 205 CHECK_TYPES_END |
| 206 } |
| 207 } |
| 208 |
| 209 |
| 210 TEST(ResetTypingInfo) { |
| 211 const char test_function[] = |
| 212 "function GeometricMean(stdlib, foreign, buffer) {\n" |
| 213 " \"use asm\";\n" |
| 214 "\n" |
| 215 " var exp = stdlib.Math.exp;\n" |
| 216 " var log = stdlib.Math.log;\n" |
| 217 " var values = new stdlib.Float64Array(buffer);\n" |
| 218 "\n" |
| 219 " function logSum(start, end) {\n" |
| 220 " start = start|0;\n" |
| 221 " end = end|0;\n" |
| 222 "\n" |
| 223 " var sum = 0.0, p = 0, q = 0;\n" |
| 224 "\n" |
| 225 " // asm.js forces byte addressing of the heap by requiring shifting " |
| 226 "by 3\n" |
| 227 " for (p = start << 3, q = end << 3; (p|0) < (q|0); p = (p + 8)|0) {\n" |
| 228 " sum = sum + +log(values[p>>3]);\n" |
| 229 " }\n" |
| 230 "\n" |
| 231 " return +sum;\n" |
| 232 " }\n" |
| 233 "\n" |
| 234 " function geometricMean(start, end) {\n" |
| 235 " start = start|0;\n" |
| 236 " end = end|0;\n" |
| 237 "\n" |
| 238 " return +exp(+logSum(start, end) / +((end - start)|0));\n" |
| 239 " }\n" |
| 240 "\n" |
| 241 " return { geometricMean: geometricMean };\n" |
| 242 "}\n"; |
| 243 |
| 244 v8::V8::Initialize(); |
| 245 HandleAndZoneScope handles; |
| 246 |
| 247 i::Isolate* isolate = CcTest::i_isolate(); |
| 248 i::Factory* factory = isolate->factory(); |
| 249 |
| 250 i::Handle<i::String> source_code = |
| 251 factory->NewStringFromUtf8(i::CStrVector(test_function)) |
| 252 .ToHandleChecked(); |
| 253 |
| 254 i::Handle<i::Script> script = factory->NewScript(source_code); |
| 255 |
| 256 i::ParseInfo info(handles.main_zone(), script); |
| 257 i::Parser parser(&info); |
| 258 parser.set_allow_harmony_arrow_functions(true); |
| 259 parser.set_allow_harmony_sloppy(true); |
| 260 info.set_global(); |
| 261 info.set_lazy(false); |
| 262 info.set_allow_lazy_parsing(false); |
| 263 info.set_toplevel(true); |
| 264 |
| 265 i::CompilationInfo compilation_info(&info); |
| 266 CHECK(i::Compiler::ParseAndAnalyze(&info)); |
| 267 info.set_literal( |
| 268 info.scope()->declarations()->at(0)->AsFunctionDeclaration()->fun()); |
| 269 |
| 270 // Core of the test. |
| 271 ZoneVector<ExpressionTypeEntry> types(handles.main_zone()); |
| 272 ExpressionTypeCollector(&compilation_info, &types).Run(); |
| 273 CheckAllSame(types, DEFAULT_TYPE); |
| 274 |
| 275 TypeSetter(&compilation_info).Run(); |
| 276 |
| 277 ExpressionTypeCollector(&compilation_info, &types).Run(); |
| 278 CheckAllSame(types, INT32_TYPE); |
| 279 |
| 280 TypingReseter(&compilation_info).Run(); |
| 281 |
| 282 ExpressionTypeCollector(&compilation_info, &types).Run(); |
| 283 CheckAllSame(types, DEFAULT_TYPE); |
| 284 } |
OLD | NEW |