Index: src/interpreter/bytecode-generator.cc |
diff --git a/src/interpreter/bytecode-generator.cc b/src/interpreter/bytecode-generator.cc |
index 53ef7d95ff187ea6dd69dd4775f526c23d788ba3..cd566c3acbf9f43f52d9cf09ed152efeffce76ca 100644 |
--- a/src/interpreter/bytecode-generator.cc |
+++ b/src/interpreter/bytecode-generator.cc |
@@ -1492,6 +1492,21 @@ void BytecodeGenerator::VisitClassLiteral(ClassLiteral* expr) { |
.LoadNamedProperty(literal, prototype_string(), feedback_index(slot)) |
.StoreAccumulatorInRegister(prototype); |
+ // The below proxy exists to allow static field initializers to have the |
+ // correct home object and receiver. It's only necessary if the initializers |
+ // are called as a part of class definition instead of immediately after it, |
+ // in which case they could simply refer to the class just constructed. The |
+ // latter is actually what's currently specified, and this will need to be |
+ // changed if that behavior is settled upon. See also |
+ // https://github.com/tc39/proposal-class-public-fields/issues/50 |
+ VariableProxy* static_initializer_proxy = expr->static_initializer_proxy(); |
+ if (static_initializer_proxy != nullptr) { |
+ Variable* variable = static_initializer_proxy->var(); |
+ builder()->LoadAccumulatorWithRegister(literal); |
+ VisitVariableAssignment(variable, Token::INIT, |
+ FeedbackVectorSlot::Invalid()); |
+ } |
+ |
VisitClassLiteralProperties(expr, literal, prototype); |
builder()->CallRuntime(Runtime::kToFastProperties, literal, 1); |
// Assign to class variable. |
@@ -1546,6 +1561,29 @@ void BytecodeGenerator::VisitClassLiteralProperties(ClassLiteral* expr, |
for (int i = 0; i < expr->properties()->length(); i++) { |
ClassLiteral::Property* property = expr->properties()->at(i); |
+ if (property->kind() == ClassLiteral::Property::FIELD && |
+ !property->is_static()) { |
+ // Non-static properties produced by the parser have as their 'key' an |
+ // expression producing their name and as their 'value' a variable which |
+ // is refered to by the synthetic initializer function in order to |
+ // determine the name during class instantiation. This is necessary |
+ // because computed names must only be evaluated once, at class definition |
+ // time. |
+ // That is, code which looks like `class C { [f()] = 1; }` is desugared |
+ // into something like |
+ // class C { constructor(){ this.[.class-field-0-name] = 1; } }; |
+ // let .class-field-0-name = f(); |
+ // except that the assignment to .class-field-name-0 occurs interleaved |
+ // with the rest of the class body; it is performed by the block in which |
+ // this comment appears. |
+ DCHECK(property->value()->IsVariableProxy()); |
+ Variable* variable = property->value()->AsVariableProxy()->var(); |
+ VisitForAccumulatorValue(property->key()); |
+ VisitVariableAssignment(variable, Token::INIT, |
+ FeedbackVectorSlot::Invalid()); |
+ continue; |
+ } |
+ |
// Set-up receiver. |
Register new_receiver = property->is_static() ? literal : prototype; |
if (new_receiver != old_receiver) { |
@@ -1594,7 +1632,12 @@ void BytecodeGenerator::VisitClassLiteralProperties(ClassLiteral* expr, |
break; |
} |
case ClassLiteral::Property::FIELD: { |
- UNREACHABLE(); |
+ DCHECK(property->is_static()); |
+ builder() |
+ ->LoadLiteral(Smi::FromInt(property->NeedsSetFunctionName())) |
+ .StoreAccumulatorInRegister(set_function_name); |
+ builder()->CallRuntime(Runtime::kDefineDataPropertyInLiteral, receiver, |
+ 5); |
break; |
} |
} |