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

Side by Side Diff: test/unittests/interpreter/register-translator-unittest.cc

Issue 1613163002: [interpreter] Wide register support. (Closed) Base URL: https://chromium.googlesource.com/v8/v8.git@master
Patch Set: Rebase Created 4 years, 10 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 | « test/unittests/interpreter/bytecodes-unittest.cc ('k') | test/unittests/unittests.gyp » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(Empty)
1 // Copyright 2014 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 <stack>
6
7 #include "src/v8.h"
8
9 #include "src/interpreter/register-translator.h"
10 #include "src/isolate.h"
11 #include "test/unittests/test-utils.h"
12
13 namespace v8 {
14 namespace internal {
15 namespace interpreter {
16
17 class RegisterTranslatorTest : public TestWithIsolateAndZone,
18 private RegisterMover {
19 public:
20 RegisterTranslatorTest() : translator_(this), move_count_(0) {
21 window_start_ =
22 RegisterTranslator::DistanceToTranslationWindow(Register(0));
23 window_width_ =
24 Register::MaxRegisterIndexForByteOperand() - window_start_ + 1;
25 }
26
27 ~RegisterTranslatorTest() override {}
28
29 bool PopMoveAndMatch(Register from, Register to) {
30 CHECK(from.is_valid() && to.is_valid());
31 const std::pair<Register, Register> top = moves_.top();
32 moves_.pop();
33 return top.first == from && top.second == to;
34 }
35
36 int move_count() const { return move_count_; }
37 RegisterTranslator* translator() { return &translator_; }
38
39 int window_start() const { return window_start_; }
40 int window_width() const { return window_width_; }
41 int window_limit() const { return window_start_ + window_width_; }
42
43 protected:
44 static const char* const kBadOperandRegex;
45
46 private:
47 void MoveRegisterUntranslated(Register from, Register to) override {
48 moves_.push(std::make_pair(from, to));
49 move_count_++;
50 }
51
52 bool RegisterOperandIsMovable(Bytecode bytecode, int operand_index) override {
53 OperandType operand_type =
54 Bytecodes::GetOperandType(bytecode, operand_index);
55 if (operand_type == OperandType::kReg8 ||
56 operand_type == OperandType::kReg16) {
57 if (operand_index == Bytecodes::NumberOfOperands(bytecode) - 1) {
58 return true;
59 }
60 OperandType next_operand_type =
61 Bytecodes::GetOperandType(bytecode, operand_index + 1);
62 return (next_operand_type != OperandType::kRegCount8 &&
63 next_operand_type != OperandType::kRegCount16);
64 } else {
65 return false;
66 }
67 }
68
69 RegisterTranslator translator_;
70 std::stack<std::pair<Register, Register>> moves_;
71 int move_count_;
72 int window_start_;
73 int window_width_;
74 };
75
76 const char* const RegisterTranslatorTest::kBadOperandRegex =
77 ".*OperandType::kReg8 && mover\\(\\)->RegisterOperandIsMovable.*";
78
79 TEST_F(RegisterTranslatorTest, TestFrameSizeAdjustmentsForTranslationWindow) {
80 EXPECT_EQ(0, RegisterTranslator::RegisterCountAdjustment(0, 0));
81 EXPECT_EQ(0, RegisterTranslator::RegisterCountAdjustment(10, 10));
82 EXPECT_EQ(window_width(),
83 RegisterTranslator::RegisterCountAdjustment(173, 0));
84 EXPECT_EQ(window_width(),
85 RegisterTranslator::RegisterCountAdjustment(173, 137));
86 EXPECT_EQ(window_width(),
87 RegisterTranslator::RegisterCountAdjustment(173, 137));
88 EXPECT_EQ(0, RegisterTranslator::RegisterCountAdjustment(0, 120));
89 EXPECT_EQ(window_limit(),
90 RegisterTranslator::RegisterCountAdjustment(0, 128));
91 EXPECT_EQ(window_limit(),
92 RegisterTranslator::RegisterCountAdjustment(0, 129));
93 EXPECT_EQ(window_limit() - 32,
94 RegisterTranslator::RegisterCountAdjustment(32, 129));
95 }
96
97 TEST_F(RegisterTranslatorTest, TestInTranslationWindow) {
98 EXPECT_GE(window_start(), 0);
99 EXPECT_FALSE(
100 RegisterTranslator::InTranslationWindow(Register(window_start() - 1)));
101 EXPECT_TRUE(RegisterTranslator::InTranslationWindow(
102 Register(Register::MaxRegisterIndexForByteOperand())));
103 EXPECT_FALSE(RegisterTranslator::InTranslationWindow(
104 Register(Register::MaxRegisterIndexForByteOperand() + 1)));
105 for (int index = window_start(); index < window_limit(); index += 1) {
106 EXPECT_TRUE(RegisterTranslator::InTranslationWindow(Register(index)));
107 }
108 }
109
110 TEST_F(RegisterTranslatorTest, FitsInReg8Operand) {
111 EXPECT_GT(window_start(), 0);
112 EXPECT_TRUE(RegisterTranslator::FitsInReg8Operand(
113 Register::FromParameterIndex(0, 3)));
114 EXPECT_TRUE(RegisterTranslator::FitsInReg8Operand(
115 Register::FromParameterIndex(2, 3)));
116 EXPECT_TRUE(RegisterTranslator::FitsInReg8Operand(Register(0)));
117 EXPECT_TRUE(
118 RegisterTranslator::FitsInReg8Operand(Register(window_start() - 1)));
119 EXPECT_FALSE(RegisterTranslator::FitsInReg8Operand(Register(kMaxInt8)));
120 EXPECT_FALSE(RegisterTranslator::FitsInReg8Operand(Register(kMaxInt8 + 1)));
121 for (int index = window_start(); index < window_limit(); index += 1) {
122 EXPECT_FALSE(RegisterTranslator::FitsInReg8Operand(Register(index)));
123 }
124 }
125
126 TEST_F(RegisterTranslatorTest, FitsInReg16Operand) {
127 EXPECT_GT(window_start(), 0);
128 EXPECT_TRUE(RegisterTranslator::FitsInReg16Operand(
129 Register::FromParameterIndex(0, 3)));
130 EXPECT_TRUE(RegisterTranslator::FitsInReg16Operand(
131 Register::FromParameterIndex(2, 3)));
132 EXPECT_TRUE(RegisterTranslator::FitsInReg16Operand(
133 Register::FromParameterIndex(0, 999)));
134 EXPECT_TRUE(RegisterTranslator::FitsInReg16Operand(
135 Register::FromParameterIndex(0, Register::MaxParameterIndex() + 1)));
136 EXPECT_TRUE(RegisterTranslator::FitsInReg16Operand(Register(0)));
137 EXPECT_TRUE(
138 RegisterTranslator::FitsInReg16Operand(Register(window_start() - 1)));
139 EXPECT_TRUE(RegisterTranslator::FitsInReg16Operand(Register(kMaxInt8 + 1)));
140 EXPECT_TRUE(RegisterTranslator::FitsInReg16Operand(Register(kMaxInt8 + 2)));
141 for (int index = 0; index <= kMaxInt16 - window_width(); index += 1) {
142 EXPECT_TRUE(RegisterTranslator::FitsInReg16Operand(Register(index)));
143 }
144 for (int index = Register::MaxRegisterIndex() - window_width() + 1;
145 index < Register::MaxRegisterIndex() + 2; index += 1) {
146 EXPECT_FALSE(RegisterTranslator::FitsInReg16Operand(Register(index)));
147 }
148 }
149
150 TEST_F(RegisterTranslatorTest, NoTranslationRequired) {
151 Register window_reg(window_start());
152 Register local_reg(57);
153 uint32_t operands[] = {local_reg.ToRawOperand()};
154 translator()->TranslateInputRegisters(Bytecode::kLdar, operands, 1);
155 translator()->TranslateOutputRegisters();
156 EXPECT_EQ(0, move_count());
157
158 Register param_reg = Register::FromParameterIndex(129, 130);
159 operands[0] = param_reg.ToRawOperand();
160 translator()->TranslateInputRegisters(Bytecode::kLdar, operands, 1);
161 translator()->TranslateOutputRegisters();
162 EXPECT_EQ(0, move_count());
163 }
164
165 TEST_F(RegisterTranslatorTest, TranslationRequired) {
166 Register window_reg(window_start());
167 Register local_reg(137);
168 Register local_reg_translated(local_reg.index() + window_width());
169
170 uint32_t operands[] = {local_reg.ToRawOperand()};
171 translator()->TranslateInputRegisters(Bytecode::kLdar, operands, 1);
172 EXPECT_EQ(1, move_count());
173 EXPECT_TRUE(PopMoveAndMatch(local_reg_translated, window_reg));
174 translator()->TranslateOutputRegisters();
175 EXPECT_EQ(2, move_count());
176 EXPECT_TRUE(PopMoveAndMatch(window_reg, local_reg_translated));
177
178 Register param_reg = Register::FromParameterIndex(0, 130);
179 operands[0] = {param_reg.ToRawOperand()};
180 translator()->TranslateInputRegisters(Bytecode::kLdar, operands, 1);
181 EXPECT_EQ(3, move_count());
182 EXPECT_TRUE(PopMoveAndMatch(param_reg, window_reg));
183 translator()->TranslateOutputRegisters();
184 EXPECT_EQ(4, move_count());
185 EXPECT_TRUE(PopMoveAndMatch(window_reg, param_reg));
186 }
187
188 TEST_F(RegisterTranslatorTest, RangeTranslation) {
189 Register window0(window_start());
190 Register window1(window_start() + 1);
191 Register window2(window_start() + 2);
192 uint32_t operands[3];
193
194 // Bytecode::kNew with valid range operand.
195 Register constructor0(0);
196 Register args0(1);
197 operands[0] = constructor0.ToRawOperand();
198 operands[1] = args0.ToRawOperand();
199 operands[2] = 1;
200 translator()->TranslateInputRegisters(Bytecode::kNew, operands, 3);
201 translator()->TranslateOutputRegisters();
202 EXPECT_EQ(0, move_count());
203
204 // Bytecode::kNewWide with valid range operand.
205 Register constructor1(128);
206 Register constructor1_translated(constructor1.index() + window_width());
207 Register args1(129);
208 Register args1_translated(args1.index() + window_width());
209 operands[0] = constructor1.ToRawOperand();
210 operands[1] = args1.ToRawOperand();
211 operands[2] = 3;
212 translator()->TranslateInputRegisters(Bytecode::kNewWide, operands, 3);
213 translator()->TranslateOutputRegisters();
214 EXPECT_EQ(0, move_count());
215 }
216
217 TEST_F(RegisterTranslatorTest, BadRange0) {
218 // Bytecode::kNew with invalid range operand (kMaybeReg8).
219 Register constructor1(128);
220 Register args1(129);
221 uint32_t operands[] = {constructor1.ToRawOperand(), args1.ToRawOperand(), 3};
222 ASSERT_DEATH_IF_SUPPORTED(
223 translator()->TranslateInputRegisters(Bytecode::kNew, operands, 3),
224 kBadOperandRegex);
225 }
226
227 TEST_F(RegisterTranslatorTest, BadRange1) {
228 // Bytecode::kForInPrepare with invalid range operand (kRegTriple8)
229 Register for_in_state(160);
230 Register for_in_state_translated(for_in_state.index() + window_width());
231 uint32_t operands[] = {for_in_state.ToRawOperand()};
232 ASSERT_DEATH_IF_SUPPORTED(translator()->TranslateInputRegisters(
233 Bytecode::kForInPrepare, operands, 1),
234 kBadOperandRegex);
235 }
236
237 TEST_F(RegisterTranslatorTest, BadRange2) {
238 // Bytecode::kForInNext with invalid range operand (kRegPair8)
239 Register receiver(192);
240 Register receiver_translated(receiver.index() + window_width());
241 Register index(193);
242 Register index_translated(index.index() + window_width());
243 Register cache_info_pair(194);
244 Register cache_info_pair_translated(cache_info_pair.index() + window_width());
245 uint32_t operands[] = {receiver.ToRawOperand(), index.ToRawOperand(),
246 cache_info_pair.ToRawOperand()};
247 ASSERT_DEATH_IF_SUPPORTED(
248 translator()->TranslateInputRegisters(Bytecode::kForInNext, operands, 3),
249 kBadOperandRegex);
250 }
251
252 } // namespace interpreter
253 } // namespace internal
254 } // namespace v8
OLDNEW
« no previous file with comments | « test/unittests/interpreter/bytecodes-unittest.cc ('k') | test/unittests/unittests.gyp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698