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

Side by Side Diff: src/interpreter/bytecode-generator.cc

Issue 2142333002: Refactor class declaration logic to the parser and runtime Base URL: https://chromium.googlesource.com/v8/v8.git@master
Patch Set: cleanup per Adam Created 4 years, 5 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
OLDNEW
1 // Copyright 2015 the V8 project authors. All rights reserved. 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 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "src/interpreter/bytecode-generator.h" 5 #include "src/interpreter/bytecode-generator.h"
6 6
7 #include "src/ast/scopes.h" 7 #include "src/ast/scopes.h"
8 #include "src/code-stubs.h" 8 #include "src/code-stubs.h"
9 #include "src/compiler.h" 9 #include "src/compiler.h"
10 #include "src/interpreter/bytecode-register-allocator.h" 10 #include "src/interpreter/bytecode-register-allocator.h"
(...skipping 1276 matching lines...) Expand 10 before | Expand all | Expand 10 after
1287 Compiler::GetSharedFunctionInfo(expr, info()->script(), info()); 1287 Compiler::GetSharedFunctionInfo(expr, info()->script(), info());
1288 if (shared_info.is_null()) { 1288 if (shared_info.is_null()) {
1289 return SetStackOverflow(); 1289 return SetStackOverflow();
1290 } 1290 }
1291 uint8_t flags = CreateClosureFlags::Encode(expr->pretenure(), 1291 uint8_t flags = CreateClosureFlags::Encode(expr->pretenure(),
1292 scope()->is_function_scope()); 1292 scope()->is_function_scope());
1293 builder()->CreateClosure(shared_info, flags); 1293 builder()->CreateClosure(shared_info, flags);
1294 execution_result()->SetResultInAccumulator(); 1294 execution_result()->SetResultInAccumulator();
1295 } 1295 }
1296 1296
1297 void BytecodeGenerator::VisitClassLiteral(ClassLiteral* expr) {
1298 if (expr->scope()->ContextLocalCount() > 0) {
1299 VisitNewLocalBlockContext(expr->scope());
1300 ContextScope scope(this, expr->scope());
1301 VisitDeclarations(expr->scope()->declarations());
1302 VisitClassLiteralContents(expr);
1303 } else {
1304 VisitDeclarations(expr->scope()->declarations());
1305 VisitClassLiteralContents(expr);
1306 }
1307 }
1308
1309 void BytecodeGenerator::VisitClassLiteralContents(ClassLiteral* expr) {
1310 VisitClassLiteralForRuntimeDefinition(expr);
1311
1312 // Load the "prototype" from the constructor.
1313 register_allocator()->PrepareForConsecutiveAllocations(2);
1314 Register literal = register_allocator()->NextConsecutiveRegister();
1315 Register prototype = register_allocator()->NextConsecutiveRegister();
1316 Handle<String> name = isolate()->factory()->prototype_string();
1317 FeedbackVectorSlot slot = expr->PrototypeSlot();
1318 builder()
1319 ->StoreAccumulatorInRegister(literal)
1320 .LoadNamedProperty(literal, name, feedback_index(slot))
1321 .StoreAccumulatorInRegister(prototype);
1322
1323 VisitClassLiteralProperties(expr, literal, prototype);
1324 builder()->CallRuntime(Runtime::kToFastProperties, literal, 1);
1325 // Assign to class variable.
1326 if (expr->class_variable_proxy() != nullptr) {
1327 Variable* var = expr->class_variable_proxy()->var();
1328 FeedbackVectorSlot slot = expr->NeedsProxySlot()
1329 ? expr->ProxySlot()
1330 : FeedbackVectorSlot::Invalid();
1331 VisitVariableAssignment(var, Token::INIT, slot);
1332 }
1333 execution_result()->SetResultInAccumulator();
1334 }
1335
1336 void BytecodeGenerator::VisitClassLiteralForRuntimeDefinition(
1337 ClassLiteral* expr) {
1338 AccumulatorResultScope result_scope(this);
1339 register_allocator()->PrepareForConsecutiveAllocations(4);
1340 Register extends = register_allocator()->NextConsecutiveRegister();
1341 Register constructor = register_allocator()->NextConsecutiveRegister();
1342 Register start_position = register_allocator()->NextConsecutiveRegister();
1343 Register end_position = register_allocator()->NextConsecutiveRegister();
1344
1345 VisitForAccumulatorValueOrTheHole(expr->extends());
1346 builder()->StoreAccumulatorInRegister(extends);
1347
1348 VisitForAccumulatorValue(expr->constructor());
1349 builder()
1350 ->StoreAccumulatorInRegister(constructor)
1351 .LoadLiteral(Smi::FromInt(expr->start_position()))
1352 .StoreAccumulatorInRegister(start_position)
1353 .LoadLiteral(Smi::FromInt(expr->end_position()))
1354 .StoreAccumulatorInRegister(end_position)
1355 .CallRuntime(Runtime::kDefineClass, extends, 4);
1356 result_scope.SetResultInAccumulator();
1357 }
1358
1359 void BytecodeGenerator::VisitClassLiteralProperties(ClassLiteral* expr,
1360 Register literal,
1361 Register prototype) {
1362 RegisterAllocationScope register_scope(this);
1363 register_allocator()->PrepareForConsecutiveAllocations(5);
1364 Register receiver = register_allocator()->NextConsecutiveRegister();
1365 Register key = register_allocator()->NextConsecutiveRegister();
1366 Register value = register_allocator()->NextConsecutiveRegister();
1367 Register attr = register_allocator()->NextConsecutiveRegister();
1368 Register set_function_name = register_allocator()->NextConsecutiveRegister();
1369
1370 bool attr_assigned = false;
1371 Register old_receiver = Register::invalid_value();
1372
1373 // Create nodes to store method values into the literal.
1374 for (int i = 0; i < expr->properties()->length(); i++) {
1375 ObjectLiteral::Property* property = expr->properties()->at(i);
1376
1377 // Set-up receiver.
1378 Register new_receiver = property->is_static() ? literal : prototype;
1379 if (new_receiver != old_receiver) {
1380 builder()->MoveRegister(new_receiver, receiver);
1381 old_receiver = new_receiver;
1382 }
1383
1384 VisitForAccumulatorValue(property->key());
1385 builder()->CastAccumulatorToName().StoreAccumulatorInRegister(key);
1386 // The static prototype property is read only. We handle the non computed
1387 // property name case in the parser. Since this is the only case where we
1388 // need to check for an own read only property we special case this so we do
1389 // not need to do this for every property.
1390 if (property->is_static() && property->is_computed_name()) {
1391 VisitClassLiteralStaticPrototypeWithComputedName(key);
1392 }
1393 VisitForAccumulatorValue(property->value());
1394 builder()->StoreAccumulatorInRegister(value);
1395
1396 VisitSetHomeObject(value, receiver, property);
1397
1398 if (!attr_assigned) {
1399 builder()
1400 ->LoadLiteral(Smi::FromInt(DONT_ENUM))
1401 .StoreAccumulatorInRegister(attr);
1402 attr_assigned = true;
1403 }
1404
1405 switch (property->kind()) {
1406 case ObjectLiteral::Property::CONSTANT:
1407 case ObjectLiteral::Property::MATERIALIZED_LITERAL:
1408 case ObjectLiteral::Property::PROTOTYPE:
1409 // Invalid properties for ES6 classes.
1410 UNREACHABLE();
1411 break;
1412 case ObjectLiteral::Property::COMPUTED: {
1413 builder()
1414 ->LoadLiteral(Smi::FromInt(property->NeedsSetFunctionName()))
1415 .StoreAccumulatorInRegister(set_function_name);
1416 builder()->CallRuntime(Runtime::kDefineDataPropertyInLiteral, receiver,
1417 5);
1418 break;
1419 }
1420 case ObjectLiteral::Property::GETTER: {
1421 builder()->CallRuntime(Runtime::kDefineGetterPropertyUnchecked,
1422 receiver, 4);
1423 break;
1424 }
1425 case ObjectLiteral::Property::SETTER: {
1426 builder()->CallRuntime(Runtime::kDefineSetterPropertyUnchecked,
1427 receiver, 4);
1428 break;
1429 }
1430 }
1431 }
1432 }
1433
1434 void BytecodeGenerator::VisitClassLiteralStaticPrototypeWithComputedName(
1435 Register key) {
1436 BytecodeLabel done;
1437 builder()
1438 ->LoadLiteral(isolate()->factory()->prototype_string())
1439 .CompareOperation(Token::Value::EQ_STRICT, key)
1440 .JumpIfFalse(&done)
1441 .CallRuntime(Runtime::kThrowStaticPrototypeError, Register(0), 0)
1442 .Bind(&done);
1443 }
1444
1445 void BytecodeGenerator::VisitNativeFunctionLiteral( 1297 void BytecodeGenerator::VisitNativeFunctionLiteral(
1446 NativeFunctionLiteral* expr) { 1298 NativeFunctionLiteral* expr) {
1447 // Find or build a shared function info for the native function template. 1299 // Find or build a shared function info for the native function template.
1448 Handle<SharedFunctionInfo> shared_info = 1300 Handle<SharedFunctionInfo> shared_info =
1449 Compiler::GetSharedFunctionInfoForNative(expr->extension(), expr->name()); 1301 Compiler::GetSharedFunctionInfoForNative(expr->extension(), expr->name());
1450 builder()->CreateClosure(shared_info, NOT_TENURED); 1302 builder()->CreateClosure(shared_info, NOT_TENURED);
1451 execution_result()->SetResultInAccumulator(); 1303 execution_result()->SetResultInAccumulator();
1452 } 1304 }
1453 1305
1454 void BytecodeGenerator::VisitDoExpression(DoExpression* expr) { 1306 void BytecodeGenerator::VisitDoExpression(DoExpression* expr) {
(...skipping 1705 matching lines...) Expand 10 before | Expand all | Expand 10 after
3160 return execution_context()->scope()->language_mode(); 3012 return execution_context()->scope()->language_mode();
3161 } 3013 }
3162 3014
3163 int BytecodeGenerator::feedback_index(FeedbackVectorSlot slot) const { 3015 int BytecodeGenerator::feedback_index(FeedbackVectorSlot slot) const {
3164 return TypeFeedbackVector::GetIndex(slot); 3016 return TypeFeedbackVector::GetIndex(slot);
3165 } 3017 }
3166 3018
3167 } // namespace interpreter 3019 } // namespace interpreter
3168 } // namespace internal 3020 } // namespace internal
3169 } // namespace v8 3021 } // namespace v8
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698