| // Copyright 2016 the V8 project authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "src/init/v8.h" |
| |
| #include "test/cctest/cctest.h" |
| |
| namespace v8 { |
| namespace internal { |
| namespace test_usecounters { |
| |
| int* global_use_counts = nullptr; |
| |
| void MockUseCounterCallback(v8::Isolate* isolate, |
| v8::Isolate::UseCounterFeature feature) { |
| ++global_use_counts[feature]; |
| } |
| |
| TEST(AssigmentExpressionLHSIsCall) { |
| v8::Isolate* isolate = CcTest::isolate(); |
| v8::HandleScope scope(isolate); |
| LocalContext env; |
| int use_counts[v8::Isolate::kUseCounterFeatureCount] = {}; |
| global_use_counts = use_counts; |
| CcTest::isolate()->SetUseCounterCallback(MockUseCounterCallback); |
| |
| // AssignmentExpressions whose LHS is not a call do not increment counters |
| CompileRun("function f(){ a = 0; a()[b] = 0; }"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy]); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict]); |
| CompileRun("function f(){ ++a; ++a()[b]; }"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy]); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict]); |
| CompileRun("function f(){ 'use strict'; a = 0; a()[b] = 0; }"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy]); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict]); |
| CompileRun("function f(){ 'use strict'; ++a; ++a()[b]; }"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy]); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict]); |
| |
| // AssignmentExpressions whose LHS is a call increment appropriate counters |
| CompileRun("function f(){ a() = 0; }"); |
| CHECK_NE(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy]); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict]); |
| use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy] = 0; |
| CompileRun("function f(){ 'use strict'; a() = 0; }"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy]); |
| CHECK_NE(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict]); |
| use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict] = 0; |
| |
| // UpdateExpressions whose LHS is a call increment appropriate counters |
| CompileRun("function f(){ ++a(); }"); |
| CHECK_NE(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy]); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict]); |
| use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy] = 0; |
| CompileRun("function f(){ 'use strict'; ++a(); }"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInSloppy]); |
| CHECK_NE(0, use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict]); |
| use_counts[v8::Isolate::kAssigmentExpressionLHSIsCallInStrict] = 0; |
| } |
| |
| TEST(RegExpMatchIsTrueishOnNonJSRegExp) { |
| v8::Isolate* isolate = CcTest::isolate(); |
| v8::HandleScope scope(isolate); |
| LocalContext env; |
| int use_counts[v8::Isolate::kUseCounterFeatureCount] = {}; |
| global_use_counts = use_counts; |
| CcTest::isolate()->SetUseCounterCallback(MockUseCounterCallback); |
| |
| CompileRun("new RegExp(/./); new RegExp('');"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kRegExpMatchIsTrueishOnNonJSRegExp]); |
| CHECK_EQ(0, use_counts[v8::Isolate::kRegExpMatchIsFalseishOnJSRegExp]); |
| |
| CompileRun("let p = { [Symbol.match]: true }; new RegExp(p);"); |
| CHECK_EQ(1, use_counts[v8::Isolate::kRegExpMatchIsTrueishOnNonJSRegExp]); |
| CHECK_EQ(0, use_counts[v8::Isolate::kRegExpMatchIsFalseishOnJSRegExp]); |
| } |
| |
| TEST(RegExpMatchIsFalseishOnJSRegExp) { |
| v8::Isolate* isolate = CcTest::isolate(); |
| v8::HandleScope scope(isolate); |
| LocalContext env; |
| int use_counts[v8::Isolate::kUseCounterFeatureCount] = {}; |
| global_use_counts = use_counts; |
| CcTest::isolate()->SetUseCounterCallback(MockUseCounterCallback); |
| |
| CompileRun("new RegExp(/./); new RegExp('');"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kRegExpMatchIsTrueishOnNonJSRegExp]); |
| CHECK_EQ(0, use_counts[v8::Isolate::kRegExpMatchIsFalseishOnJSRegExp]); |
| |
| CompileRun("let p = /./; p[Symbol.match] = false; new RegExp(p);"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kRegExpMatchIsTrueishOnNonJSRegExp]); |
| CHECK_EQ(1, use_counts[v8::Isolate::kRegExpMatchIsFalseishOnJSRegExp]); |
| } |
| |
| TEST(ObjectPrototypeHasElements) { |
| v8::Isolate* isolate = CcTest::isolate(); |
| v8::HandleScope scope(isolate); |
| LocalContext env; |
| int use_counts[v8::Isolate::kUseCounterFeatureCount] = {}; |
| global_use_counts = use_counts; |
| CcTest::isolate()->SetUseCounterCallback(MockUseCounterCallback); |
| |
| CompileRun("var o = {}; o[1] = 2;"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kObjectPrototypeHasElements]); |
| |
| CompileRun("var o = {}; var p = {}; o.__proto__ = p; p[1] = 2;"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kObjectPrototypeHasElements]); |
| |
| CompileRun("Object.prototype[1] = 2;"); |
| CHECK_EQ(1, use_counts[v8::Isolate::kObjectPrototypeHasElements]); |
| } |
| |
| TEST(ArrayPrototypeHasElements) { |
| v8::Isolate* isolate = CcTest::isolate(); |
| v8::HandleScope scope(isolate); |
| LocalContext env; |
| int use_counts[v8::Isolate::kUseCounterFeatureCount] = {}; |
| global_use_counts = use_counts; |
| CcTest::isolate()->SetUseCounterCallback(MockUseCounterCallback); |
| |
| CompileRun("var a = []; a[1] = 2;"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kArrayPrototypeHasElements]); |
| |
| CompileRun("var a = []; var p = []; a.__proto__ = p; p[1] = 2;"); |
| CHECK_EQ(0, use_counts[v8::Isolate::kArrayPrototypeHasElements]); |
| |
| CompileRun("Array.prototype[1] = 2;"); |
| CHECK_EQ(1, use_counts[v8::Isolate::kArrayPrototypeHasElements]); |
| } |
| |
| } // namespace test_usecounters |
| } // namespace internal |
| } // namespace v8 |