Commit 00364877 authored by oth's avatar oth Committed by Commit bot

[Interpreter] Add tests for adding strings and heap number arithmetic.

BUG=v8:4280
LOG=N

Review URL: https://codereview.chromium.org/1374613002

Cr-Commit-Position: refs/heads/master@{#30986}
parent df7df4df
...@@ -349,109 +349,148 @@ TEST(InterpreterLoadStoreRegisters) { ...@@ -349,109 +349,148 @@ TEST(InterpreterLoadStoreRegisters) {
} }
TEST(InterpreterAdd) { static const Token::Value kArithmeticOperators[] = {
HandleAndZoneScope handles; Token::Value::ADD, Token::Value::SUB, Token::Value::MUL, Token::Value::DIV,
// TODO(rmcilroy): Do add tests for heap numbers and strings once we support Token::Value::MOD};
// them.
BytecodeArrayBuilder builder(handles.main_isolate(), handles.main_zone());
builder.set_locals_count(1);
builder.set_parameter_count(1);
Register reg(0);
builder.LoadLiteral(Smi::FromInt(1))
.StoreAccumulatorInRegister(reg)
.LoadLiteral(Smi::FromInt(2))
.BinaryOperation(Token::Value::ADD, reg)
.Return();
Handle<BytecodeArray> bytecode_array = builder.ToBytecodeArray();
InterpreterTester tester(handles.main_isolate(), bytecode_array);
auto callable = tester.GetCallable<>();
Handle<Object> return_val = callable().ToHandleChecked();
CHECK_EQ(Smi::cast(*return_val), Smi::FromInt(3));
}
TEST(InterpreterSub) {
HandleAndZoneScope handles;
// TODO(rmcilroy): Do add tests for heap numbers once we support them.
BytecodeArrayBuilder builder(handles.main_isolate(), handles.main_zone());
builder.set_locals_count(1);
builder.set_parameter_count(1);
Register reg(0);
builder.LoadLiteral(Smi::FromInt(5))
.StoreAccumulatorInRegister(reg)
.LoadLiteral(Smi::FromInt(31))
.BinaryOperation(Token::Value::SUB, reg)
.Return();
Handle<BytecodeArray> bytecode_array = builder.ToBytecodeArray();
InterpreterTester tester(handles.main_isolate(), bytecode_array); static double BinaryOpC(Token::Value op, double lhs, double rhs) {
auto callable = tester.GetCallable<>(); switch (op) {
Handle<Object> return_val = callable().ToHandleChecked(); case Token::Value::ADD:
CHECK_EQ(Smi::cast(*return_val), Smi::FromInt(-26)); return lhs + rhs;
case Token::Value::SUB:
return lhs - rhs;
case Token::Value::MUL:
return lhs * rhs;
case Token::Value::DIV:
return lhs / rhs;
case Token::Value::MOD:
return std::fmod(lhs, rhs);
default:
UNREACHABLE();
return std::numeric_limits<double>::min();
}
} }
TEST(InterpreterMul) { TEST(InterpreterBinaryOpsSmi) {
HandleAndZoneScope handles; int lhs_inputs[] = {3266, 1024, 0, -17, -18000};
// TODO(rmcilroy): Do add tests for heap numbers once we support them. int rhs_inputs[] = {3266, 5, 4, 3, 2, 1, -1, -2};
BytecodeArrayBuilder builder(handles.main_isolate(), handles.main_zone()); for (size_t l = 0; l < arraysize(lhs_inputs); l++) {
builder.set_locals_count(1); for (size_t r = 0; r < arraysize(rhs_inputs); r++) {
builder.set_parameter_count(1); for (size_t o = 0; o < arraysize(kArithmeticOperators); o++) {
Register reg(0); HandleAndZoneScope handles;
builder.LoadLiteral(Smi::FromInt(111)) i::Factory* factory = handles.main_isolate()->factory();
.StoreAccumulatorInRegister(reg) BytecodeArrayBuilder builder(handles.main_isolate(),
.LoadLiteral(Smi::FromInt(6)) handles.main_zone());
.BinaryOperation(Token::Value::MUL, reg) builder.set_locals_count(1);
.Return(); builder.set_parameter_count(1);
Handle<BytecodeArray> bytecode_array = builder.ToBytecodeArray(); Register reg(0);
int lhs = lhs_inputs[l];
int rhs = rhs_inputs[l];
builder.LoadLiteral(Smi::FromInt(lhs))
.StoreAccumulatorInRegister(reg)
.LoadLiteral(Smi::FromInt(rhs))
.BinaryOperation(kArithmeticOperators[o], reg)
.Return();
Handle<BytecodeArray> bytecode_array = builder.ToBytecodeArray();
InterpreterTester tester(handles.main_isolate(), bytecode_array); InterpreterTester tester(handles.main_isolate(), bytecode_array);
auto callable = tester.GetCallable<>(); auto callable = tester.GetCallable<>();
Handle<Object> return_val = callable().ToHandleChecked(); Handle<Object> return_value = callable().ToHandleChecked();
CHECK_EQ(Smi::cast(*return_val), Smi::FromInt(666)); Handle<Object> expected_value =
factory->NewNumber(BinaryOpC(kArithmeticOperators[o], lhs, rhs));
CHECK(return_value->SameValue(*expected_value));
}
}
}
} }
TEST(InterpreterDiv) { TEST(InterpreterBinaryOpsHeapNumber) {
HandleAndZoneScope handles; double lhs_inputs[] = {3266.101, 1024.12, 0.01, -17.99, -18000.833, 9.1e17};
// TODO(rmcilroy): Do add tests for heap numbers once we support them. double rhs_inputs[] = {3266.101, 5.999, 4.778, 3.331, 2.643,
BytecodeArrayBuilder builder(handles.main_isolate(), handles.main_zone()); 1.1, -1.8, -2.9, 8.3e-27};
builder.set_locals_count(1); for (size_t l = 0; l < arraysize(lhs_inputs); l++) {
builder.set_parameter_count(1); for (size_t r = 0; r < arraysize(rhs_inputs); r++) {
Register reg(0); for (size_t o = 0; o < arraysize(kArithmeticOperators); o++) {
builder.LoadLiteral(Smi::FromInt(-20)) HandleAndZoneScope handles;
.StoreAccumulatorInRegister(reg) i::Factory* factory = handles.main_isolate()->factory();
.LoadLiteral(Smi::FromInt(5)) BytecodeArrayBuilder builder(handles.main_isolate(),
.BinaryOperation(Token::Value::DIV, reg) handles.main_zone());
.Return(); builder.set_locals_count(1);
Handle<BytecodeArray> bytecode_array = builder.ToBytecodeArray(); builder.set_parameter_count(1);
Register reg(0);
double lhs = lhs_inputs[l];
double rhs = rhs_inputs[l];
builder.LoadLiteral(factory->NewNumber(lhs))
.StoreAccumulatorInRegister(reg)
.LoadLiteral(factory->NewNumber(rhs))
.BinaryOperation(kArithmeticOperators[o], reg)
.Return();
Handle<BytecodeArray> bytecode_array = builder.ToBytecodeArray();
InterpreterTester tester(handles.main_isolate(), bytecode_array); InterpreterTester tester(handles.main_isolate(), bytecode_array);
auto callable = tester.GetCallable<>(); auto callable = tester.GetCallable<>();
Handle<Object> return_val = callable().ToHandleChecked(); Handle<Object> return_value = callable().ToHandleChecked();
CHECK_EQ(Smi::cast(*return_val), Smi::FromInt(-4)); Handle<Object> expected_value =
factory->NewNumber(BinaryOpC(kArithmeticOperators[o], lhs, rhs));
CHECK(return_value->SameValue(*expected_value));
}
}
}
} }
TEST(InterpreterMod) { TEST(InterpreterStringAdd) {
HandleAndZoneScope handles; HandleAndZoneScope handles;
// TODO(rmcilroy): Do add tests for heap numbers once we support them. i::Factory* factory = handles.main_isolate()->factory();
BytecodeArrayBuilder builder(handles.main_isolate(), handles.main_zone());
builder.set_locals_count(1);
builder.set_parameter_count(1);
Register reg(0);
builder.LoadLiteral(Smi::FromInt(121))
.StoreAccumulatorInRegister(reg)
.LoadLiteral(Smi::FromInt(100))
.BinaryOperation(Token::Value::MOD, reg)
.Return();
Handle<BytecodeArray> bytecode_array = builder.ToBytecodeArray();
InterpreterTester tester(handles.main_isolate(), bytecode_array); struct TestCase {
auto callable = tester.GetCallable<>(); Handle<Object> lhs;
Handle<Object> return_val = callable().ToHandleChecked(); Handle<Object> rhs;
CHECK_EQ(Smi::cast(*return_val), Smi::FromInt(21)); Handle<Object> expected_value;
} test_cases[] = {
{factory->NewStringFromStaticChars("a"),
factory->NewStringFromStaticChars("b"),
factory->NewStringFromStaticChars("ab")},
{factory->NewStringFromStaticChars("aaaaaa"),
factory->NewStringFromStaticChars("b"),
factory->NewStringFromStaticChars("aaaaaab")},
{factory->NewStringFromStaticChars("aaa"),
factory->NewStringFromStaticChars("bbbbb"),
factory->NewStringFromStaticChars("aaabbbbb")},
{factory->NewStringFromStaticChars(""),
factory->NewStringFromStaticChars("b"),
factory->NewStringFromStaticChars("b")},
{factory->NewStringFromStaticChars("a"),
factory->NewStringFromStaticChars(""),
factory->NewStringFromStaticChars("a")},
{factory->NewStringFromStaticChars("1.11"), factory->NewHeapNumber(2.5),
factory->NewStringFromStaticChars("1.112.5")},
{factory->NewStringFromStaticChars("-1.11"), factory->NewHeapNumber(2.56),
factory->NewStringFromStaticChars("-1.112.56")},
{factory->NewStringFromStaticChars(""), factory->NewHeapNumber(2.5),
factory->NewStringFromStaticChars("2.5")},
};
for (size_t i = 0; i < arraysize(test_cases); i++) {
BytecodeArrayBuilder builder(handles.main_isolate(), handles.main_zone());
builder.set_locals_count(1);
builder.set_parameter_count(1);
Register reg(0);
builder.LoadLiteral(test_cases[i].lhs)
.StoreAccumulatorInRegister(reg)
.LoadLiteral(test_cases[i].rhs)
.BinaryOperation(Token::Value::ADD, reg)
.Return();
Handle<BytecodeArray> bytecode_array = builder.ToBytecodeArray();
InterpreterTester tester(handles.main_isolate(), bytecode_array);
auto callable = tester.GetCallable<>();
Handle<Object> return_value = callable().ToHandleChecked();
CHECK(return_value->SameValue(*test_cases[i].expected_value));
}
} }
......
...@@ -166,6 +166,7 @@ TEST_F(RuntimeInterpreterTest, ToBoolean) { ...@@ -166,6 +166,7 @@ TEST_F(RuntimeInterpreterTest, ToBoolean) {
} }
} }
} // Namespace interpreter } // Namespace interpreter
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment