mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-25 18:34:14 +00:00 
			
		
		
		
	LibJS: Make Value() default-construct the undefined value
The special empty value (that we use for array holes, Optional<Value> when empty and a few other other placeholder/sentinel tasks) still exists, but you now create one via JS::js_special_empty_value() and check for it with Value::is_special_empty_value(). The main idea here is to make it very unlikely to accidentally create an unexpected special empty value.
This commit is contained in:
		
							parent
							
								
									0d91363742
								
							
						
					
					
						commit
						3cf50539ec
					
				
				
				Notes:
				
					github-actions[bot]
				
				2025-04-05 09:21:31 +00:00 
				
			
			Author: https://github.com/awesomekling
Commit: 3cf50539ec
Pull-request: https://github.com/LadybirdBrowser/ladybird/pull/4232
			
					 43 changed files with 165 additions and 122 deletions
				
			
		
							
								
								
									
										34
									
								
								AK/Vector.h
									
										
									
									
									
								
							
							
						
						
									
										34
									
								
								AK/Vector.h
									
										
									
									
									
								
							|  | @ -686,6 +686,22 @@ public: | ||||||
|         return {}; |         return {}; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     ErrorOr<void> try_resize_with_default_value(size_t new_size, T const& default_value, bool keep_capacity) | ||||||
|  |     requires(!contains_reference) | ||||||
|  |     { | ||||||
|  |         if (new_size <= size()) { | ||||||
|  |             shrink(new_size, keep_capacity); | ||||||
|  |             return {}; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         TRY(try_ensure_capacity(new_size)); | ||||||
|  | 
 | ||||||
|  |         for (size_t i = size(); i < new_size; ++i) | ||||||
|  |             new (slot(i)) StorageType { default_value }; | ||||||
|  |         m_size = new_size; | ||||||
|  |         return {}; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     ErrorOr<void> try_resize_and_keep_capacity(size_t new_size) |     ErrorOr<void> try_resize_and_keep_capacity(size_t new_size) | ||||||
|     requires(!contains_reference) |     requires(!contains_reference) | ||||||
|     { |     { | ||||||
|  | @ -733,6 +749,24 @@ public: | ||||||
|         MUST(try_resize_and_keep_capacity(new_size)); |         MUST(try_resize_and_keep_capacity(new_size)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     void resize_with_default_value_and_keep_capacity(size_t new_size, T const& default_value) | ||||||
|  |     requires(!contains_reference) | ||||||
|  |     { | ||||||
|  |         MUST(try_resize_with_default_value(new_size, default_value, true)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void resize_with_default_value(size_t new_size, T const& default_value, bool keep_capacity = false) | ||||||
|  |     requires(!contains_reference) | ||||||
|  |     { | ||||||
|  |         MUST(try_resize_with_default_value(new_size, default_value, keep_capacity)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void fill(T const& value) | ||||||
|  |     { | ||||||
|  |         for (size_t i = 0; i < size(); ++i) | ||||||
|  |             at(i) = value; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     void shrink_to_fit() |     void shrink_to_fit() | ||||||
|     { |     { | ||||||
|         if (size() == capacity()) |         if (size() == capacity()) | ||||||
|  |  | ||||||
|  | @ -137,7 +137,7 @@ static ThrowCompletionOr<ClassElementName> class_key_to_property_name(VM& vm, Ex | ||||||
|         return ClassElementName { private_environment->resolve_private_identifier(private_identifier.string()) }; |         return ClassElementName { private_environment->resolve_private_identifier(private_identifier.string()) }; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     VERIFY(!prop_key.is_empty()); |     VERIFY(!prop_key.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     if (prop_key.is_object()) |     if (prop_key.is_object()) | ||||||
|         prop_key = TRY(prop_key.to_primitive(vm, Value::PreferredType::String)); |         prop_key = TRY(prop_key.to_primitive(vm, Value::PreferredType::String)); | ||||||
|  |  | ||||||
|  | @ -1201,7 +1201,7 @@ Bytecode::CodeGenerationErrorOr<Optional<ScopedOperand>> ArrayExpression::genera | ||||||
|         // If all elements are constant primitives, we can just emit a single instruction to initialize the array,
 |         // If all elements are constant primitives, we can just emit a single instruction to initialize the array,
 | ||||||
|         // instead of emitting instructions to manually evaluate them one-by-one
 |         // instead of emitting instructions to manually evaluate them one-by-one
 | ||||||
|         Vector<Value> values; |         Vector<Value> values; | ||||||
|         values.resize(m_elements.size()); |         values.resize_with_default_value(m_elements.size(), js_special_empty_value()); | ||||||
|         for (auto i = 0u; i < m_elements.size(); ++i) { |         for (auto i = 0u; i < m_elements.size(); ++i) { | ||||||
|             if (!m_elements[i]) |             if (!m_elements[i]) | ||||||
|                 continue; |                 continue; | ||||||
|  | @ -1221,7 +1221,7 @@ Bytecode::CodeGenerationErrorOr<Optional<ScopedOperand>> ArrayExpression::genera | ||||||
|             auto value = TRY((*it)->generate_bytecode(generator)).value(); |             auto value = TRY((*it)->generate_bytecode(generator)).value(); | ||||||
|             args.append(generator.copy_if_needed_to_preserve_evaluation_order(value)); |             args.append(generator.copy_if_needed_to_preserve_evaluation_order(value)); | ||||||
|         } else { |         } else { | ||||||
|             args.append(generator.add_constant(Value())); |             args.append(generator.add_constant(js_special_empty_value())); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -1235,7 +1235,7 @@ Bytecode::CodeGenerationErrorOr<Optional<ScopedOperand>> ArrayExpression::genera | ||||||
|     if (first_spread != m_elements.end()) { |     if (first_spread != m_elements.end()) { | ||||||
|         for (auto it = first_spread; it != m_elements.end(); ++it) { |         for (auto it = first_spread; it != m_elements.end(); ++it) { | ||||||
|             if (!*it) { |             if (!*it) { | ||||||
|                 generator.emit<Bytecode::Op::ArrayAppend>(dst, generator.add_constant(Value()), false); |                 generator.emit<Bytecode::Op::ArrayAppend>(dst, generator.add_constant(js_special_empty_value()), false); | ||||||
|             } else { |             } else { | ||||||
|                 auto value = TRY((*it)->generate_bytecode(generator)).value(); |                 auto value = TRY((*it)->generate_bytecode(generator)).value(); | ||||||
|                 generator.emit<Bytecode::Op::ArrayAppend>(dst, value, *it && is<SpreadExpression>(**it)); |                 generator.emit<Bytecode::Op::ArrayAppend>(dst, value, *it && is<SpreadExpression>(**it)); | ||||||
|  |  | ||||||
|  | @ -1304,7 +1304,7 @@ ScopedOperand Generator::add_constant(Value value) | ||||||
|             m_null_constant = append_new_constant(); |             m_null_constant = append_new_constant(); | ||||||
|         return m_null_constant.value(); |         return m_null_constant.value(); | ||||||
|     } |     } | ||||||
|     if (value.is_empty()) { |     if (value.is_special_empty_value()) { | ||||||
|         if (!m_empty_constant.has_value()) |         if (!m_empty_constant.has_value()) | ||||||
|             m_empty_constant = append_new_constant(); |             m_empty_constant = append_new_constant(); | ||||||
|         return m_empty_constant.value(); |         return m_empty_constant.value(); | ||||||
|  |  | ||||||
|  | @ -298,7 +298,7 @@ public: | ||||||
|                 emit<Bytecode::Op::PrepareYield>(Operand(Register::saved_return_value()), value); |                 emit<Bytecode::Op::PrepareYield>(Operand(Register::saved_return_value()), value); | ||||||
|             else |             else | ||||||
|                 emit<Bytecode::Op::Mov>(Operand(Register::saved_return_value()), value); |                 emit<Bytecode::Op::Mov>(Operand(Register::saved_return_value()), value); | ||||||
|             emit<Bytecode::Op::Mov>(Operand(Register::exception()), add_constant(Value {})); |             emit<Bytecode::Op::Mov>(Operand(Register::exception()), add_constant(js_special_empty_value())); | ||||||
|             emit<Bytecode::Op::Jump>(Label { *m_current_basic_block->finalizer() }); |             emit<Bytecode::Op::Jump>(Label { *m_current_basic_block->finalizer() }); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -86,7 +86,7 @@ static ByteString format_operand(StringView name, Operand operand, Bytecode::Exe | ||||||
|     case Operand::Type::Constant: { |     case Operand::Type::Constant: { | ||||||
|         builder.append("\033[36m"sv); |         builder.append("\033[36m"sv); | ||||||
|         auto value = executable.constants[operand.index() - executable.number_of_registers]; |         auto value = executable.constants[operand.index() - executable.number_of_registers]; | ||||||
|         if (value.is_empty()) |         if (value.is_special_empty_value()) | ||||||
|             builder.append("<Empty>"sv); |             builder.append("<Empty>"sv); | ||||||
|         else if (value.is_boolean()) |         else if (value.is_boolean()) | ||||||
|             builder.appendff("Bool({})", value.as_bool() ? "true"sv : "false"sv); |             builder.appendff("Bool({})", value.as_bool() ? "true"sv : "false"sv); | ||||||
|  | @ -268,12 +268,13 @@ ThrowCompletionOr<Value> Interpreter::run(Script& script_record, GC::Ptr<Environ | ||||||
|             auto result_or_error = run_executable(*executable, {}, {}); |             auto result_or_error = run_executable(*executable, {}, {}); | ||||||
|             if (result_or_error.value.is_error()) |             if (result_or_error.value.is_error()) | ||||||
|                 result = result_or_error.value.release_error(); |                 result = result_or_error.value.release_error(); | ||||||
|             else |             else { | ||||||
|                 result = result_or_error.return_register_value.value_or(js_undefined()); |                 result = result_or_error.return_register_value.is_special_empty_value() ? normal_completion(js_undefined()) : result_or_error.return_register_value; | ||||||
|  |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // b. If result is a normal completion and result.[[Value]] is empty, then
 |         // b. If result is a normal completion and result.[[Value]] is empty, then
 | ||||||
|         if (result.type() == Completion::Type::Normal && result.value().is_empty()) { |         if (result.type() == Completion::Type::Normal && result.value().is_special_empty_value()) { | ||||||
|             // i. Set result to NormalCompletion(undefined).
 |             // i. Set result to NormalCompletion(undefined).
 | ||||||
|             result = normal_completion(js_undefined()); |             result = normal_completion(js_undefined()); | ||||||
|         } |         } | ||||||
|  | @ -300,9 +301,7 @@ ThrowCompletionOr<Value> Interpreter::run(Script& script_record, GC::Ptr<Environ | ||||||
|     // 17. Return ? result.
 |     // 17. Return ? result.
 | ||||||
|     if (result.is_abrupt()) { |     if (result.is_abrupt()) { | ||||||
|         VERIFY(result.type() == Completion::Type::Throw); |         VERIFY(result.type() == Completion::Type::Throw); | ||||||
|         auto error = result.release_error(); |         return result.release_error(); | ||||||
|         VERIFY(!error.value().is_empty()); |  | ||||||
|         return error; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return result.value(); |     return result.value(); | ||||||
|  | @ -515,12 +514,12 @@ FLATTEN_ON_CLANG void Interpreter::run_bytecode(size_t entry_point) | ||||||
| 
 | 
 | ||||||
|         handle_ContinuePendingUnwind: { |         handle_ContinuePendingUnwind: { | ||||||
|             auto& instruction = *reinterpret_cast<Op::ContinuePendingUnwind const*>(&bytecode[program_counter]); |             auto& instruction = *reinterpret_cast<Op::ContinuePendingUnwind const*>(&bytecode[program_counter]); | ||||||
|             if (auto exception = reg(Register::exception()); !exception.is_empty()) { |             if (auto exception = reg(Register::exception()); !exception.is_special_empty_value()) { | ||||||
|                 if (handle_exception(program_counter, exception) == HandleExceptionResponse::ExitFromExecutable) |                 if (handle_exception(program_counter, exception) == HandleExceptionResponse::ExitFromExecutable) | ||||||
|                     return; |                     return; | ||||||
|                 goto start; |                 goto start; | ||||||
|             } |             } | ||||||
|             if (!saved_return_value().is_empty()) { |             if (!saved_return_value().is_special_empty_value()) { | ||||||
|                 do_return(saved_return_value()); |                 do_return(saved_return_value()); | ||||||
|                 if (auto handlers = executable.exception_handlers_for_offset(program_counter); handlers.has_value()) { |                 if (auto handlers = executable.exception_handlers_for_offset(program_counter); handlers.has_value()) { | ||||||
|                     if (auto finalizer = handlers.value().finalizer_offset; finalizer.has_value()) { |                     if (auto finalizer = handlers.value().finalizer_offset; finalizer.has_value()) { | ||||||
|  | @ -528,7 +527,7 @@ FLATTEN_ON_CLANG void Interpreter::run_bytecode(size_t entry_point) | ||||||
|                         auto& unwind_context = running_execution_context.unwind_contexts.last(); |                         auto& unwind_context = running_execution_context.unwind_contexts.last(); | ||||||
|                         VERIFY(unwind_context.executable == m_current_executable); |                         VERIFY(unwind_context.executable == m_current_executable); | ||||||
|                         reg(Register::saved_return_value()) = reg(Register::return_value()); |                         reg(Register::saved_return_value()) = reg(Register::return_value()); | ||||||
|                         reg(Register::return_value()) = {}; |                         reg(Register::return_value()) = js_special_empty_value(); | ||||||
|                         program_counter = finalizer.value(); |                         program_counter = finalizer.value(); | ||||||
|                         // the unwind_context will be pop'ed when entering the finally block
 |                         // the unwind_context will be pop'ed when entering the finally block
 | ||||||
|                         goto start; |                         goto start; | ||||||
|  | @ -734,21 +733,21 @@ Interpreter::ResultAndReturnRegister Interpreter::run_executable(Executable& exe | ||||||
|     auto& running_execution_context = vm().running_execution_context(); |     auto& running_execution_context = vm().running_execution_context(); | ||||||
|     u32 registers_and_constants_and_locals_count = executable.number_of_registers + executable.constants.size() + executable.local_variable_names.size(); |     u32 registers_and_constants_and_locals_count = executable.number_of_registers + executable.constants.size() + executable.local_variable_names.size(); | ||||||
|     if (running_execution_context.registers_and_constants_and_locals.size() < registers_and_constants_and_locals_count) |     if (running_execution_context.registers_and_constants_and_locals.size() < registers_and_constants_and_locals_count) | ||||||
|         running_execution_context.registers_and_constants_and_locals.resize(registers_and_constants_and_locals_count); |         running_execution_context.registers_and_constants_and_locals.resize_with_default_value(registers_and_constants_and_locals_count, js_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     TemporaryChange restore_running_execution_context { m_running_execution_context, &running_execution_context }; |     TemporaryChange restore_running_execution_context { m_running_execution_context, &running_execution_context }; | ||||||
|     TemporaryChange restore_arguments { m_arguments, running_execution_context.arguments.span() }; |     TemporaryChange restore_arguments { m_arguments, running_execution_context.arguments.span() }; | ||||||
|     TemporaryChange restore_registers_and_constants_and_locals { m_registers_and_constants_and_locals, running_execution_context.registers_and_constants_and_locals.span() }; |     TemporaryChange restore_registers_and_constants_and_locals { m_registers_and_constants_and_locals, running_execution_context.registers_and_constants_and_locals.span() }; | ||||||
| 
 | 
 | ||||||
|     reg(Register::accumulator()) = initial_accumulator_value; |     reg(Register::accumulator()) = initial_accumulator_value; | ||||||
|     reg(Register::return_value()) = {}; |     reg(Register::return_value()) = js_special_empty_value(); | ||||||
| 
 | 
 | ||||||
|     // NOTE: We only copy the `this` value from ExecutionContext if it's not already set.
 |     // NOTE: We only copy the `this` value from ExecutionContext if it's not already set.
 | ||||||
|     //       If we are re-entering an async/generator context, the `this` value
 |     //       If we are re-entering an async/generator context, the `this` value
 | ||||||
|     //       may have already been cached by a ResolveThisBinding instruction,
 |     //       may have already been cached by a ResolveThisBinding instruction,
 | ||||||
|     //       and subsequent instructions expect this value to be set.
 |     //       and subsequent instructions expect this value to be set.
 | ||||||
|     if (reg(Register::this_value()).is_empty()) |     if (reg(Register::this_value()).is_special_empty_value()) | ||||||
|         reg(Register::this_value()) = running_execution_context.this_value; |         reg(Register::this_value()) = running_execution_context.this_value.value_or(js_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     running_execution_context.executable = &executable; |     running_execution_context.executable = &executable; | ||||||
| 
 | 
 | ||||||
|  | @ -764,7 +763,7 @@ Interpreter::ResultAndReturnRegister Interpreter::run_executable(Executable& exe | ||||||
|         auto const& registers_and_constants_and_locals = running_execution_context.registers_and_constants_and_locals; |         auto const& registers_and_constants_and_locals = running_execution_context.registers_and_constants_and_locals; | ||||||
|         for (size_t i = 0; i < executable.number_of_registers; ++i) { |         for (size_t i = 0; i < executable.number_of_registers; ++i) { | ||||||
|             String value_string; |             String value_string; | ||||||
|             if (registers_and_constants_and_locals[i].is_empty()) |             if (registers_and_constants_and_locals[i].is_special_empty_value()) | ||||||
|                 value_string = "(empty)"_string; |                 value_string = "(empty)"_string; | ||||||
|             else |             else | ||||||
|                 value_string = registers_and_constants_and_locals[i].to_string_without_side_effects(); |                 value_string = registers_and_constants_and_locals[i].to_string_without_side_effects(); | ||||||
|  | @ -773,14 +772,14 @@ Interpreter::ResultAndReturnRegister Interpreter::run_executable(Executable& exe | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     auto return_value = js_undefined(); |     auto return_value = js_undefined(); | ||||||
|     if (!reg(Register::return_value()).is_empty()) |     if (!reg(Register::return_value()).is_special_empty_value()) | ||||||
|         return_value = reg(Register::return_value()); |         return_value = reg(Register::return_value()); | ||||||
|     auto exception = reg(Register::exception()); |     auto exception = reg(Register::exception()); | ||||||
| 
 | 
 | ||||||
|     vm().run_queued_promise_jobs(); |     vm().run_queued_promise_jobs(); | ||||||
|     vm().finish_execution_generation(); |     vm().finish_execution_generation(); | ||||||
| 
 | 
 | ||||||
|     if (!exception.is_empty()) |     if (!exception.is_special_empty_value()) | ||||||
|         return { throw_completion(exception), running_execution_context.registers_and_constants_and_locals[0] }; |         return { throw_completion(exception), running_execution_context.registers_and_constants_and_locals[0] }; | ||||||
|     return { return_value, running_execution_context.registers_and_constants_and_locals[0] }; |     return { return_value, running_execution_context.registers_and_constants_and_locals[0] }; | ||||||
| } | } | ||||||
|  | @ -802,7 +801,7 @@ void Interpreter::leave_unwind_context() | ||||||
| void Interpreter::catch_exception(Operand dst) | void Interpreter::catch_exception(Operand dst) | ||||||
| { | { | ||||||
|     set(dst, reg(Register::exception())); |     set(dst, reg(Register::exception())); | ||||||
|     reg(Register::exception()) = {}; |     reg(Register::exception()) = js_special_empty_value(); | ||||||
|     auto& context = running_execution_context().unwind_contexts.last(); |     auto& context = running_execution_context().unwind_contexts.last(); | ||||||
|     VERIFY(!context.handler_called); |     VERIFY(!context.handler_called); | ||||||
|     VERIFY(context.executable == ¤t_executable()); |     VERIFY(context.executable == ¤t_executable()); | ||||||
|  | @ -817,7 +816,7 @@ void Interpreter::restore_scheduled_jump() | ||||||
| 
 | 
 | ||||||
| void Interpreter::leave_finally() | void Interpreter::leave_finally() | ||||||
| { | { | ||||||
|     reg(Register::exception()) = {}; |     reg(Register::exception()) = js_special_empty_value(); | ||||||
|     m_scheduled_jump = running_execution_context().previously_scheduled_jumps.take_last(); |     m_scheduled_jump = running_execution_context().previously_scheduled_jumps.take_last(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -1265,7 +1264,7 @@ inline ThrowCompletionOr<Value> perform_call(Interpreter& interpreter, Value thi | ||||||
|     Value return_value; |     Value return_value; | ||||||
|     if (call_type == Op::CallType::DirectEval) { |     if (call_type == Op::CallType::DirectEval) { | ||||||
|         if (callee == interpreter.realm().intrinsics().eval_function()) |         if (callee == interpreter.realm().intrinsics().eval_function()) | ||||||
|             return_value = TRY(perform_eval(vm, !argument_values.is_empty() ? argument_values[0].value_or(JS::js_undefined()) : js_undefined(), vm.in_strict_mode() ? CallerMode::Strict : CallerMode::NonStrict, EvalMode::Direct)); |             return_value = TRY(perform_eval(vm, !argument_values.is_empty() ? argument_values[0] : js_undefined(), vm.in_strict_mode() ? CallerMode::Strict : CallerMode::NonStrict, EvalMode::Direct)); | ||||||
|         else |         else | ||||||
|             return_value = TRY(JS::call(vm, function, this_value, argument_values)); |             return_value = TRY(JS::call(vm, function, this_value, argument_values)); | ||||||
|     } else if (call_type == Op::CallType::Call) |     } else if (call_type == Op::CallType::Call) | ||||||
|  | @ -2539,13 +2538,13 @@ ThrowCompletionOr<void> DeleteByIdWithThis::execute_impl(Bytecode::Interpreter& | ||||||
| ThrowCompletionOr<void> ResolveThisBinding::execute_impl(Bytecode::Interpreter& interpreter) const | ThrowCompletionOr<void> ResolveThisBinding::execute_impl(Bytecode::Interpreter& interpreter) const | ||||||
| { | { | ||||||
|     auto& cached_this_value = interpreter.reg(Register::this_value()); |     auto& cached_this_value = interpreter.reg(Register::this_value()); | ||||||
|     if (!cached_this_value.is_empty()) |     if (!cached_this_value.is_special_empty_value()) | ||||||
|         return {}; |         return {}; | ||||||
|     // OPTIMIZATION: Because the value of 'this' cannot be reassigned during a function execution, it's
 |     // OPTIMIZATION: Because the value of 'this' cannot be reassigned during a function execution, it's
 | ||||||
|     //               resolved once and then saved for subsequent use.
 |     //               resolved once and then saved for subsequent use.
 | ||||||
|     auto& running_execution_context = interpreter.running_execution_context(); |     auto& running_execution_context = interpreter.running_execution_context(); | ||||||
|     if (auto function = running_execution_context.function; function && is<ECMAScriptFunctionObject>(*function) && !static_cast<ECMAScriptFunctionObject&>(*function).allocates_function_environment()) { |     if (auto function = running_execution_context.function; function && is<ECMAScriptFunctionObject>(*function) && !static_cast<ECMAScriptFunctionObject&>(*function).allocates_function_environment()) { | ||||||
|         cached_this_value = running_execution_context.this_value; |         cached_this_value = running_execution_context.this_value.value(); | ||||||
|     } else { |     } else { | ||||||
|         auto& vm = interpreter.vm(); |         auto& vm = interpreter.vm(); | ||||||
|         cached_this_value = TRY(vm.resolve_this_binding()); |         cached_this_value = TRY(vm.resolve_this_binding()); | ||||||
|  | @ -2801,7 +2800,7 @@ ThrowCompletionOr<void> ThrowIfTDZ::execute_impl(Bytecode::Interpreter& interpre | ||||||
| { | { | ||||||
|     auto& vm = interpreter.vm(); |     auto& vm = interpreter.vm(); | ||||||
|     auto value = interpreter.get(m_src); |     auto value = interpreter.get(m_src); | ||||||
|     if (value.is_empty()) |     if (value.is_special_empty_value()) | ||||||
|         return vm.throw_completion<ReferenceError>(ErrorType::BindingNotInitialized, value.to_string_without_side_effects()); |         return vm.throw_completion<ReferenceError>(ErrorType::BindingNotInitialized, value.to_string_without_side_effects()); | ||||||
|     return {}; |     return {}; | ||||||
| } | } | ||||||
|  | @ -2825,20 +2824,20 @@ void LeaveUnwindContext::execute_impl(Bytecode::Interpreter& interpreter) const | ||||||
| 
 | 
 | ||||||
| void Yield::execute_impl(Bytecode::Interpreter& interpreter) const | void Yield::execute_impl(Bytecode::Interpreter& interpreter) const | ||||||
| { | { | ||||||
|     auto yielded_value = interpreter.get(m_value).value_or(js_undefined()); |     auto yielded_value = interpreter.get(m_value).is_special_empty_value() ? js_undefined() : interpreter.get(m_value); | ||||||
|     interpreter.do_return( |     interpreter.do_return( | ||||||
|         interpreter.do_yield(yielded_value, m_continuation_label)); |         interpreter.do_yield(yielded_value, m_continuation_label)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void PrepareYield::execute_impl(Bytecode::Interpreter& interpreter) const | void PrepareYield::execute_impl(Bytecode::Interpreter& interpreter) const | ||||||
| { | { | ||||||
|     auto value = interpreter.get(m_value).value_or(js_undefined()); |     auto value = interpreter.get(m_value).is_special_empty_value() ? js_undefined() : interpreter.get(m_value); | ||||||
|     interpreter.set(m_dest, interpreter.do_yield(value, {})); |     interpreter.set(m_dest, interpreter.do_yield(value, {})); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Await::execute_impl(Bytecode::Interpreter& interpreter) const | void Await::execute_impl(Bytecode::Interpreter& interpreter) const | ||||||
| { | { | ||||||
|     auto yielded_value = interpreter.get(m_argument).value_or(js_undefined()); |     auto yielded_value = interpreter.get(m_argument).is_special_empty_value() ? js_undefined() : interpreter.get(m_argument); | ||||||
|     // FIXME: If we get a pointer, which is not accurately representable as a double
 |     // FIXME: If we get a pointer, which is not accurately representable as a double
 | ||||||
|     //        will cause this to explode
 |     //        will cause this to explode
 | ||||||
|     auto continuation_value = Value(m_continuation_label.address()); |     auto continuation_value = Value(m_continuation_label.address()); | ||||||
|  | @ -3832,7 +3831,7 @@ ByteString Dump::to_byte_string_impl(Bytecode::Executable const& executable) con | ||||||
| void GetCompletionFields::execute_impl(Bytecode::Interpreter& interpreter) const | void GetCompletionFields::execute_impl(Bytecode::Interpreter& interpreter) const | ||||||
| { | { | ||||||
|     auto const& completion_cell = static_cast<CompletionCell const&>(interpreter.get(m_completion).as_cell()); |     auto const& completion_cell = static_cast<CompletionCell const&>(interpreter.get(m_completion).as_cell()); | ||||||
|     interpreter.set(m_value_dst, completion_cell.completion().value().value_or(js_undefined())); |     interpreter.set(m_value_dst, completion_cell.completion().value()); | ||||||
|     interpreter.set(m_type_dst, Value(to_underlying(completion_cell.completion().type()))); |     interpreter.set(m_type_dst, Value(to_underlying(completion_cell.completion().type()))); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -33,7 +33,7 @@ public: | ||||||
|     ThrowCompletionOr<Value> run(Script&, GC::Ptr<Environment> lexical_environment_override = nullptr); |     ThrowCompletionOr<Value> run(Script&, GC::Ptr<Environment> lexical_environment_override = nullptr); | ||||||
|     ThrowCompletionOr<Value> run(SourceTextModule&); |     ThrowCompletionOr<Value> run(SourceTextModule&); | ||||||
| 
 | 
 | ||||||
|     ThrowCompletionOr<Value> run(Bytecode::Executable& executable, Optional<size_t> entry_point = {}, Value initial_accumulator_value = {}) |     ThrowCompletionOr<Value> run(Bytecode::Executable& executable, Optional<size_t> entry_point = {}, Value initial_accumulator_value = js_special_empty_value()) | ||||||
|     { |     { | ||||||
|         auto result_and_return_register = run_executable(executable, entry_point, initial_accumulator_value); |         auto result_and_return_register = run_executable(executable, entry_point, initial_accumulator_value); | ||||||
|         return move(result_and_return_register.value); |         return move(result_and_return_register.value); | ||||||
|  | @ -43,7 +43,7 @@ public: | ||||||
|         ThrowCompletionOr<Value> value; |         ThrowCompletionOr<Value> value; | ||||||
|         Value return_register_value; |         Value return_register_value; | ||||||
|     }; |     }; | ||||||
|     ResultAndReturnRegister run_executable(Bytecode::Executable&, Optional<size_t> entry_point, Value initial_accumulator_value = {}); |     ResultAndReturnRegister run_executable(Bytecode::Executable&, Optional<size_t> entry_point, Value initial_accumulator_value = js_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     ALWAYS_INLINE Value& accumulator() { return reg(Register::accumulator()); } |     ALWAYS_INLINE Value& accumulator() { return reg(Register::accumulator()); } | ||||||
|     ALWAYS_INLINE Value& saved_return_value() { return reg(Register::saved_return_value()); } |     ALWAYS_INLINE Value& saved_return_value() { return reg(Register::saved_return_value()); } | ||||||
|  | @ -63,7 +63,7 @@ public: | ||||||
|     void do_return(Value value) |     void do_return(Value value) | ||||||
|     { |     { | ||||||
|         reg(Register::return_value()) = value; |         reg(Register::return_value()) = value; | ||||||
|         reg(Register::exception()) = {}; |         reg(Register::exception()) = js_special_empty_value(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void enter_unwind_context(); |     void enter_unwind_context(); | ||||||
|  |  | ||||||
|  | @ -46,7 +46,7 @@ ErrorOr<String> MarkupGenerator::html_from_error(Error const& object, bool in_pr | ||||||
| 
 | 
 | ||||||
| ErrorOr<void> MarkupGenerator::value_to_html(Value value, StringBuilder& output_html, HashTable<Object*>& seen_objects) | ErrorOr<void> MarkupGenerator::value_to_html(Value value, StringBuilder& output_html, HashTable<Object*>& seen_objects) | ||||||
| { | { | ||||||
|     if (value.is_empty()) { |     if (value.is_special_empty_value()) { | ||||||
|         TRY(output_html.try_append("<empty>"sv)); |         TRY(output_html.try_append("<empty>"sv)); | ||||||
|         return {}; |         return {}; | ||||||
|     } |     } | ||||||
|  | @ -166,8 +166,8 @@ ErrorOr<void> MarkupGenerator::trace_to_html(TracebackFrame const& traceback_fra | ||||||
| ErrorOr<void> MarkupGenerator::error_to_html(Error const& error, StringBuilder& html_output, bool in_promise) | ErrorOr<void> MarkupGenerator::error_to_html(Error const& error, StringBuilder& html_output, bool in_promise) | ||||||
| { | { | ||||||
|     auto& vm = error.vm(); |     auto& vm = error.vm(); | ||||||
|     auto name = error.get_without_side_effects(vm.names.name).value_or(js_undefined()); |     auto name = error.get_without_side_effects(vm.names.name); | ||||||
|     auto message = error.get_without_side_effects(vm.names.message).value_or(js_undefined()); |     auto message = error.get_without_side_effects(vm.names.message); | ||||||
|     auto name_string = name.to_string_without_side_effects(); |     auto name_string = name.to_string_without_side_effects(); | ||||||
|     auto message_string = message.to_string_without_side_effects(); |     auto message_string = message.to_string_without_side_effects(); | ||||||
|     auto uncaught_message = TRY(String::formatted("Uncaught {}[{}]: ", in_promise ? "(in promise) " : "", name_string)); |     auto uncaught_message = TRY(String::formatted("Uncaught {}[{}]: ", in_promise ? "(in promise) " : "", name_string)); | ||||||
|  |  | ||||||
|  | @ -272,8 +272,8 @@ ErrorOr<void> print_date(JS::PrintContext& print_context, JS::Date const& date, | ||||||
| 
 | 
 | ||||||
| ErrorOr<void> print_error(JS::PrintContext& print_context, JS::Object const& object, HashTable<JS::Object*>& seen_objects) | ErrorOr<void> print_error(JS::PrintContext& print_context, JS::Object const& object, HashTable<JS::Object*>& seen_objects) | ||||||
| { | { | ||||||
|     auto name = object.get_without_side_effects(print_context.vm.names.name).value_or(JS::js_undefined()); |     auto name = object.get_without_side_effects(print_context.vm.names.name); | ||||||
|     auto message = object.get_without_side_effects(print_context.vm.names.message).value_or(JS::js_undefined()); |     auto message = object.get_without_side_effects(print_context.vm.names.message); | ||||||
|     if (name.is_accessor() || message.is_accessor()) { |     if (name.is_accessor() || message.is_accessor()) { | ||||||
|         TRY(print_value(print_context, &object, seen_objects)); |         TRY(print_value(print_context, &object, seen_objects)); | ||||||
|     } else { |     } else { | ||||||
|  | @ -917,7 +917,7 @@ ErrorOr<void> print_string_object(JS::PrintContext& print_context, JS::StringObj | ||||||
| 
 | 
 | ||||||
| ErrorOr<void> print_value(JS::PrintContext& print_context, JS::Value value, HashTable<JS::Object*>& seen_objects) | ErrorOr<void> print_value(JS::PrintContext& print_context, JS::Value value, HashTable<JS::Object*>& seen_objects) | ||||||
| { | { | ||||||
|     if (value.is_empty()) { |     if (value.is_special_empty_value()) { | ||||||
|         TRY(js_out(print_context, "\033[34;1m<empty>\033[0m")); |         TRY(js_out(print_context, "\033[34;1m<empty>\033[0m")); | ||||||
|         return {}; |         return {}; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -699,9 +699,7 @@ ThrowCompletionOr<Value> perform_eval(VM& vm, Value x, CallerMode strict_caller, | ||||||
|     if (result_or_error.value.is_error()) |     if (result_or_error.value.is_error()) | ||||||
|         return result_or_error.value.release_error(); |         return result_or_error.value.release_error(); | ||||||
| 
 | 
 | ||||||
|     auto& result = result_or_error.return_register_value; |     eval_result = result_or_error.return_register_value; | ||||||
|     if (!result.is_empty()) |  | ||||||
|         eval_result = result; |  | ||||||
| 
 | 
 | ||||||
|     // 30. If result.[[Type]] is normal and result.[[Value]] is empty, then
 |     // 30. If result.[[Type]] is normal and result.[[Value]] is empty, then
 | ||||||
|     //     a. Set result to NormalCompletion(undefined).
 |     //     a. Set result to NormalCompletion(undefined).
 | ||||||
|  |  | ||||||
|  | @ -602,7 +602,7 @@ JS_DEFINE_NATIVE_FUNCTION(ArrayPrototype::find_last_index) | ||||||
| // 23.1.3.13.1 FlattenIntoArray ( target, source, sourceLen, start, depth [ , mapperFunction [ , thisArg ] ] ), https://tc39.es/ecma262/#sec-flattenintoarray
 | // 23.1.3.13.1 FlattenIntoArray ( target, source, sourceLen, start, depth [ , mapperFunction [ , thisArg ] ] ), https://tc39.es/ecma262/#sec-flattenintoarray
 | ||||||
| static ThrowCompletionOr<size_t> flatten_into_array(VM& vm, Object& new_array, Object& array, size_t array_length, size_t target_index, double depth, FunctionObject* mapper_func = {}, Value this_arg = {}) | static ThrowCompletionOr<size_t> flatten_into_array(VM& vm, Object& new_array, Object& array, size_t array_length, size_t target_index, double depth, FunctionObject* mapper_func = {}, Value this_arg = {}) | ||||||
| { | { | ||||||
|     VERIFY(!mapper_func || (!this_arg.is_empty() && depth == 1)); |     VERIFY(!mapper_func || (!this_arg.is_special_empty_value() && depth == 1)); | ||||||
| 
 | 
 | ||||||
|     for (size_t j = 0; j < array_length; ++j) { |     for (size_t j = 0; j < array_length; ++j) { | ||||||
|         auto value_exists = TRY(array.has_property(j)); |         auto value_exists = TRY(array.has_property(j)); | ||||||
|  | @ -1258,7 +1258,7 @@ JS_DEFINE_NATIVE_FUNCTION(ArrayPrototype::slice) | ||||||
| 
 | 
 | ||||||
|     double relative_end; |     double relative_end; | ||||||
| 
 | 
 | ||||||
|     if (vm.argument(1).is_undefined() || vm.argument(1).is_empty()) |     if (vm.argument(1).is_undefined()) | ||||||
|         relative_end = (double)initial_length; |         relative_end = (double)initial_length; | ||||||
|     else |     else | ||||||
|         relative_end = TRY(vm.argument(1).to_integer_or_infinity(vm)); |         relative_end = TRY(vm.argument(1).to_integer_or_infinity(vm)); | ||||||
|  |  | ||||||
|  | @ -166,7 +166,7 @@ void AsyncFunctionDriverWrapper::continue_async_execution(VM& vm, Value value, b | ||||||
|     }(); |     }(); | ||||||
| 
 | 
 | ||||||
|     if (result.is_throw_completion()) { |     if (result.is_throw_completion()) { | ||||||
|         m_top_level_promise->reject(result.throw_completion().value().value_or(js_undefined())); |         m_top_level_promise->reject(result.throw_completion().value()); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -157,7 +157,7 @@ void AsyncGenerator::execute(VM& vm, Completion completion) | ||||||
|         auto generated_value = [](Value value) -> Value { |         auto generated_value = [](Value value) -> Value { | ||||||
|             if (value.is_cell()) |             if (value.is_cell()) | ||||||
|                 return static_cast<GeneratorResult const&>(value.as_cell()).result(); |                 return static_cast<GeneratorResult const&>(value.as_cell()).result(); | ||||||
|             return value.is_empty() ? js_undefined() : value; |             return value.is_special_empty_value() ? js_undefined() : value; | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         auto generated_continuation = [&](Value value) -> Optional<size_t> { |         auto generated_continuation = [&](Value value) -> Optional<size_t> { | ||||||
|  |  | ||||||
|  | @ -65,7 +65,7 @@ public: | ||||||
|         , m_value(value) |         , m_value(value) | ||||||
|     { |     { | ||||||
|         VERIFY(type != Type::Empty); |         VERIFY(type != Type::Empty); | ||||||
|         VERIFY(!value.is_empty()); |         VERIFY(!value.is_special_empty_value()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     Completion(ThrowCompletionOr<Value> const&); |     Completion(ThrowCompletionOr<Value> const&); | ||||||
|  | @ -245,7 +245,7 @@ public: | ||||||
|         : m_value_or_error(move(value)) |         : m_value_or_error(move(value)) | ||||||
|     { |     { | ||||||
|         if constexpr (IsSame<ValueType, Value>) |         if constexpr (IsSame<ValueType, Value>) | ||||||
|             VERIFY(!m_value_or_error.template get<ValueType>().is_empty()); |             VERIFY(!m_value_or_error.template get<ValueType>().is_special_empty_value()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     ALWAYS_INLINE ThrowCompletionOr(ThrowCompletionOr const&) = default; |     ALWAYS_INLINE ThrowCompletionOr(ThrowCompletionOr const&) = default; | ||||||
|  |  | ||||||
|  | @ -793,7 +793,7 @@ void async_block_start(VM& vm, T const& async_body, PromiseCapability const& pro | ||||||
|         return; |         return; | ||||||
| 
 | 
 | ||||||
|     // 5. Resume the suspended evaluation of asyncContext. Let result be the value returned by the resumed computation.
 |     // 5. Resume the suspended evaluation of asyncContext. Let result be the value returned by the resumed computation.
 | ||||||
|     auto result = call(vm, *closure, async_context.this_value.is_empty() ? js_undefined() : async_context.this_value); |     auto result = call(vm, *closure, *async_context.this_value); | ||||||
| 
 | 
 | ||||||
|     // 6. Assert: When we return here, asyncContext has already been removed from the execution context stack and runningContext is the currently running execution context.
 |     // 6. Assert: When we return here, asyncContext has already been removed from the execution context stack and runningContext is the currently running execution context.
 | ||||||
|     VERIFY(&vm.running_execution_context() == &running_context); |     VERIFY(&vm.running_execution_context() == &running_context); | ||||||
|  | @ -828,7 +828,7 @@ Completion ECMAScriptFunctionObject::ordinary_call_evaluate_body() | ||||||
|         m_bytecode_executable = m_ecmascript_code->bytecode_executable(); |         m_bytecode_executable = m_ecmascript_code->bytecode_executable(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     vm.running_execution_context().registers_and_constants_and_locals.resize(m_local_variables_names.size() + m_bytecode_executable->number_of_registers + m_bytecode_executable->constants.size()); |     vm.running_execution_context().registers_and_constants_and_locals.resize_with_default_value(m_local_variables_names.size() + m_bytecode_executable->number_of_registers + m_bytecode_executable->constants.size(), js_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     auto result_and_frame = vm.bytecode_interpreter().run_executable(*m_bytecode_executable, {}); |     auto result_and_frame = vm.bytecode_interpreter().run_executable(*m_bytecode_executable, {}); | ||||||
| 
 | 
 | ||||||
|  | @ -840,7 +840,7 @@ Completion ECMAScriptFunctionObject::ordinary_call_evaluate_body() | ||||||
|     // NOTE: Running the bytecode should eventually return a completion.
 |     // NOTE: Running the bytecode should eventually return a completion.
 | ||||||
|     // Until it does, we assume "return" and include the undefined fallback from the call site.
 |     // Until it does, we assume "return" and include the undefined fallback from the call site.
 | ||||||
|     if (m_kind == FunctionKind::Normal) |     if (m_kind == FunctionKind::Normal) | ||||||
|         return { Completion::Type::Return, result.value_or(js_undefined()) }; |         return { Completion::Type::Return, result }; | ||||||
| 
 | 
 | ||||||
|     if (m_kind == FunctionKind::AsyncGenerator) { |     if (m_kind == FunctionKind::AsyncGenerator) { | ||||||
|         auto async_generator_object = TRY(AsyncGenerator::create(realm, result, this, vm.running_execution_context().copy())); |         auto async_generator_object = TRY(AsyncGenerator::create(realm, result, this, vm.running_execution_context().copy())); | ||||||
|  |  | ||||||
|  | @ -82,7 +82,8 @@ void ExecutionContext::visit_edges(Cell::Visitor& visitor) | ||||||
|     visitor.visit(lexical_environment); |     visitor.visit(lexical_environment); | ||||||
|     visitor.visit(private_environment); |     visitor.visit(private_environment); | ||||||
|     visitor.visit(context_owner); |     visitor.visit(context_owner); | ||||||
|     visitor.visit(this_value); |     if (this_value.has_value()) | ||||||
|  |         visitor.visit(*this_value); | ||||||
|     visitor.visit(executable); |     visitor.visit(executable); | ||||||
|     visitor.visit(function_name); |     visitor.visit(function_name); | ||||||
|     visitor.visit(arguments); |     visitor.visit(arguments); | ||||||
|  |  | ||||||
|  | @ -62,7 +62,7 @@ public: | ||||||
|     mutable RefPtr<CachedSourceRange> cached_source_range; |     mutable RefPtr<CachedSourceRange> cached_source_range; | ||||||
| 
 | 
 | ||||||
|     GC::Ptr<PrimitiveString> function_name; |     GC::Ptr<PrimitiveString> function_name; | ||||||
|     Value this_value; |     Optional<Value> this_value; | ||||||
| 
 | 
 | ||||||
|     GC::Ptr<Bytecode::Executable> executable; |     GC::Ptr<Bytecode::Executable> executable; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -21,7 +21,7 @@ FinalizationRegistry::FinalizationRegistry(Realm& realm, GC::Ref<JobCallback> cl | ||||||
| 
 | 
 | ||||||
| void FinalizationRegistry::add_finalization_record(Cell& target, Value held_value, Cell* unregister_token) | void FinalizationRegistry::add_finalization_record(Cell& target, Value held_value, Cell* unregister_token) | ||||||
| { | { | ||||||
|     VERIFY(!held_value.is_empty()); |     VERIFY(!held_value.is_special_empty_value()); | ||||||
|     m_records.append({ &target, held_value, unregister_token }); |     m_records.append({ &target, held_value, unregister_token }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -79,7 +79,7 @@ ThrowCompletionOr<Value> FunctionEnvironment::get_this_binding(VM& vm) const | ||||||
| // 9.1.1.3.1 BindThisValue ( V ), https://tc39.es/ecma262/#sec-bindthisvalue
 | // 9.1.1.3.1 BindThisValue ( V ), https://tc39.es/ecma262/#sec-bindthisvalue
 | ||||||
| ThrowCompletionOr<Value> FunctionEnvironment::bind_this_value(VM& vm, Value this_value) | ThrowCompletionOr<Value> FunctionEnvironment::bind_this_value(VM& vm, Value this_value) | ||||||
| { | { | ||||||
|     VERIFY(!this_value.is_empty()); |     VERIFY(!this_value.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     // 1. Assert: envRec.[[ThisBindingStatus]] is not lexical.
 |     // 1. Assert: envRec.[[ThisBindingStatus]] is not lexical.
 | ||||||
|     VERIFY(m_this_binding_status != ThisBindingStatus::Lexical); |     VERIFY(m_this_binding_status != ThisBindingStatus::Lexical); | ||||||
|  |  | ||||||
|  | @ -34,7 +34,7 @@ public: | ||||||
|     Value new_target() const { return m_new_target; } |     Value new_target() const { return m_new_target; } | ||||||
|     void set_new_target(Value new_target) |     void set_new_target(Value new_target) | ||||||
|     { |     { | ||||||
|         VERIFY(!new_target.is_empty()); |         VERIFY(!new_target.is_special_empty_value()); | ||||||
|         m_new_target = new_target; |         m_new_target = new_target; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -85,7 +85,7 @@ ThrowCompletionOr<Value> GeneratorObject::execute(VM& vm, Completion const& comp | ||||||
|     auto generated_value = [](Value value) -> Value { |     auto generated_value = [](Value value) -> Value { | ||||||
|         if (value.is_cell()) |         if (value.is_cell()) | ||||||
|             return static_cast<GeneratorResult const&>(value.as_cell()).result(); |             return static_cast<GeneratorResult const&>(value.as_cell()).result(); | ||||||
|         return value.is_empty() ? js_undefined() : value; |         return value.is_special_empty_value() ? js_undefined() : value; | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     auto generated_continuation = [&](Value value) -> Optional<size_t> { |     auto generated_continuation = [&](Value value) -> Optional<size_t> { | ||||||
|  |  | ||||||
|  | @ -36,10 +36,10 @@ void SimpleIndexedPropertyStorage::grow_storage_if_needed() | ||||||
|         return; |         return; | ||||||
| 
 | 
 | ||||||
|     if (m_array_size <= m_packed_elements.capacity()) { |     if (m_array_size <= m_packed_elements.capacity()) { | ||||||
|         m_packed_elements.resize_and_keep_capacity(m_array_size); |         m_packed_elements.resize_with_default_value_and_keep_capacity(m_array_size, js_special_empty_value()); | ||||||
|     } else { |     } else { | ||||||
|         // When the array is actually full grow storage by 25% at a time.
 |         // When the array is actually full grow storage by 25% at a time.
 | ||||||
|         m_packed_elements.resize_and_keep_capacity(m_array_size + (m_array_size / 4)); |         m_packed_elements.resize_with_default_value_and_keep_capacity(m_array_size + (m_array_size / 4), js_special_empty_value()); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -57,7 +57,7 @@ void SimpleIndexedPropertyStorage::put(u32 index, Value value, PropertyAttribute | ||||||
| void SimpleIndexedPropertyStorage::remove(u32 index) | void SimpleIndexedPropertyStorage::remove(u32 index) | ||||||
| { | { | ||||||
|     VERIFY(index < m_array_size); |     VERIFY(index < m_array_size); | ||||||
|     m_packed_elements[index] = {}; |     m_packed_elements[index] = js_special_empty_value(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| ValueAndAttributes SimpleIndexedPropertyStorage::take_first() | ValueAndAttributes SimpleIndexedPropertyStorage::take_first() | ||||||
|  | @ -70,14 +70,14 @@ ValueAndAttributes SimpleIndexedPropertyStorage::take_last() | ||||||
| { | { | ||||||
|     m_array_size--; |     m_array_size--; | ||||||
|     auto last_element = m_packed_elements[m_array_size]; |     auto last_element = m_packed_elements[m_array_size]; | ||||||
|     m_packed_elements[m_array_size] = {}; |     m_packed_elements[m_array_size] = js_special_empty_value(); | ||||||
|     return { last_element, default_attributes }; |     return { last_element, default_attributes }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool SimpleIndexedPropertyStorage::set_array_like_size(size_t new_size) | bool SimpleIndexedPropertyStorage::set_array_like_size(size_t new_size) | ||||||
| { | { | ||||||
|     m_array_size = new_size; |     m_array_size = new_size; | ||||||
|     m_packed_elements.resize_and_keep_capacity(new_size); |     m_packed_elements.resize_with_default_value_and_keep_capacity(new_size, js_special_empty_value()); | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -87,7 +87,7 @@ GenericIndexedPropertyStorage::GenericIndexedPropertyStorage(SimpleIndexedProper | ||||||
|     m_array_size = storage.array_like_size(); |     m_array_size = storage.array_like_size(); | ||||||
|     for (size_t i = 0; i < storage.m_packed_elements.size(); ++i) { |     for (size_t i = 0; i < storage.m_packed_elements.size(); ++i) { | ||||||
|         auto value = storage.m_packed_elements[i]; |         auto value = storage.m_packed_elements[i]; | ||||||
|         if (!value.is_empty()) |         if (!value.is_special_empty_value()) | ||||||
|             m_sparse_elements.set(i, { value, default_attributes }); |             m_sparse_elements.set(i, { value, default_attributes }); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -270,7 +270,7 @@ size_t IndexedProperties::real_size() const | ||||||
|         auto& packed_elements = static_cast<SimpleIndexedPropertyStorage const&>(*m_storage).elements(); |         auto& packed_elements = static_cast<SimpleIndexedPropertyStorage const&>(*m_storage).elements(); | ||||||
|         size_t size = 0; |         size_t size = 0; | ||||||
|         for (auto& element : packed_elements) { |         for (auto& element : packed_elements) { | ||||||
|             if (!element.is_empty()) |             if (!element.is_special_empty_value()) | ||||||
|                 ++size; |                 ++size; | ||||||
|         } |         } | ||||||
|         return size; |         return size; | ||||||
|  | @ -288,7 +288,7 @@ Vector<u32> IndexedProperties::indices() const | ||||||
|         Vector<u32> indices; |         Vector<u32> indices; | ||||||
|         indices.ensure_capacity(storage.array_like_size()); |         indices.ensure_capacity(storage.array_like_size()); | ||||||
|         for (size_t i = 0; i < elements.size(); ++i) { |         for (size_t i = 0; i < elements.size(); ++i) { | ||||||
|             if (!elements.at(i).is_empty()) |             if (!elements.at(i).is_special_empty_value()) | ||||||
|                 indices.unchecked_append(i); |                 indices.unchecked_append(i); | ||||||
|         } |         } | ||||||
|         return indices; |         return indices; | ||||||
|  |  | ||||||
|  | @ -80,7 +80,7 @@ public: | ||||||
| 
 | 
 | ||||||
|     [[nodiscard]] bool inline_has_index(u32 index) const |     [[nodiscard]] bool inline_has_index(u32 index) const | ||||||
|     { |     { | ||||||
|         return index < m_array_size && !m_packed_elements.data()[index].is_empty(); |         return index < m_array_size && !m_packed_elements.data()[index].is_special_empty_value(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [[nodiscard]] Optional<ValueAndAttributes> inline_get(u32 index) const |     [[nodiscard]] Optional<ValueAndAttributes> inline_get(u32 index) const | ||||||
|  |  | ||||||
|  | @ -128,7 +128,7 @@ ThrowCompletionOr<void> Object::set(PropertyKey const& property_key, Value value | ||||||
| { | { | ||||||
|     auto& vm = this->vm(); |     auto& vm = this->vm(); | ||||||
| 
 | 
 | ||||||
|     VERIFY(!value.is_empty()); |     VERIFY(!value.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     // 1. Let success be ? O.[[Set]](P, V, O).
 |     // 1. Let success be ? O.[[Set]](P, V, O).
 | ||||||
|     auto success = TRY(internal_set(property_key, value, this)); |     auto success = TRY(internal_set(property_key, value, this)); | ||||||
|  | @ -161,7 +161,7 @@ ThrowCompletionOr<bool> Object::create_data_property(PropertyKey const& property | ||||||
| // 7.3.6 CreateMethodProperty ( O, P, V ), https://tc39.es/ecma262/#sec-createmethodproperty
 | // 7.3.6 CreateMethodProperty ( O, P, V ), https://tc39.es/ecma262/#sec-createmethodproperty
 | ||||||
| void Object::create_method_property(PropertyKey const& property_key, Value value) | void Object::create_method_property(PropertyKey const& property_key, Value value) | ||||||
| { | { | ||||||
|     VERIFY(!value.is_empty()); |     VERIFY(!value.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     // 1. Assert: O is an ordinary, extensible object with no non-configurable properties.
 |     // 1. Assert: O is an ordinary, extensible object with no non-configurable properties.
 | ||||||
| 
 | 
 | ||||||
|  | @ -184,7 +184,7 @@ ThrowCompletionOr<bool> Object::create_data_property_or_throw(PropertyKey const& | ||||||
| { | { | ||||||
|     auto& vm = this->vm(); |     auto& vm = this->vm(); | ||||||
| 
 | 
 | ||||||
|     VERIFY(!value.is_empty()); |     VERIFY(!value.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     // 1. Let success be ? CreateDataProperty(O, P, V).
 |     // 1. Let success be ? CreateDataProperty(O, P, V).
 | ||||||
|     auto success = TRY(create_data_property(property_key, value)); |     auto success = TRY(create_data_property(property_key, value)); | ||||||
|  | @ -202,7 +202,7 @@ ThrowCompletionOr<bool> Object::create_data_property_or_throw(PropertyKey const& | ||||||
| // 7.3.8 CreateNonEnumerableDataPropertyOrThrow ( O, P, V ), https://tc39.es/ecma262/#sec-createnonenumerabledatapropertyorthrow
 | // 7.3.8 CreateNonEnumerableDataPropertyOrThrow ( O, P, V ), https://tc39.es/ecma262/#sec-createnonenumerabledatapropertyorthrow
 | ||||||
| void Object::create_non_enumerable_data_property_or_throw(PropertyKey const& property_key, Value value) | void Object::create_non_enumerable_data_property_or_throw(PropertyKey const& property_key, Value value) | ||||||
| { | { | ||||||
|     VERIFY(!value.is_empty()); |     VERIFY(!value.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     // 1. Assert: O is an ordinary, extensible object with no non-configurable properties.
 |     // 1. Assert: O is an ordinary, extensible object with no non-configurable properties.
 | ||||||
| 
 | 
 | ||||||
|  | @ -812,7 +812,7 @@ ThrowCompletionOr<Optional<PropertyDescriptor>> Object::internal_get_own_propert | ||||||
|     // 4. If X is a data property, then
 |     // 4. If X is a data property, then
 | ||||||
|     if (!value.is_accessor()) { |     if (!value.is_accessor()) { | ||||||
|         // a. Set D.[[Value]] to the value of X's [[Value]] attribute.
 |         // a. Set D.[[Value]] to the value of X's [[Value]] attribute.
 | ||||||
|         descriptor.value = value.value_or(js_undefined()); |         descriptor.value = value; | ||||||
| 
 | 
 | ||||||
|         // b. Set D.[[Writable]] to the value of X's [[Writable]] attribute.
 |         // b. Set D.[[Writable]] to the value of X's [[Writable]] attribute.
 | ||||||
|         descriptor.writable = attributes.is_writable(); |         descriptor.writable = attributes.is_writable(); | ||||||
|  | @ -883,7 +883,7 @@ ThrowCompletionOr<bool> Object::internal_has_property(PropertyKey const& propert | ||||||
| // 10.1.8.1 OrdinaryGet ( O, P, Receiver ), https://tc39.es/ecma262/#sec-ordinaryget
 | // 10.1.8.1 OrdinaryGet ( O, P, Receiver ), https://tc39.es/ecma262/#sec-ordinaryget
 | ||||||
| ThrowCompletionOr<Value> Object::internal_get(PropertyKey const& property_key, Value receiver, CacheablePropertyMetadata* cacheable_metadata, PropertyLookupPhase phase) const | ThrowCompletionOr<Value> Object::internal_get(PropertyKey const& property_key, Value receiver, CacheablePropertyMetadata* cacheable_metadata, PropertyLookupPhase phase) const | ||||||
| { | { | ||||||
|     VERIFY(!receiver.is_empty()); |     VERIFY(!receiver.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     auto& vm = this->vm(); |     auto& vm = this->vm(); | ||||||
| 
 | 
 | ||||||
|  | @ -950,8 +950,8 @@ ThrowCompletionOr<Value> Object::internal_get(PropertyKey const& property_key, V | ||||||
| // 10.1.9.1 OrdinarySet ( O, P, V, Receiver ), https://tc39.es/ecma262/#sec-ordinaryset
 | // 10.1.9.1 OrdinarySet ( O, P, V, Receiver ), https://tc39.es/ecma262/#sec-ordinaryset
 | ||||||
| ThrowCompletionOr<bool> Object::internal_set(PropertyKey const& property_key, Value value, Value receiver, CacheablePropertyMetadata* cacheable_metadata) | ThrowCompletionOr<bool> Object::internal_set(PropertyKey const& property_key, Value value, Value receiver, CacheablePropertyMetadata* cacheable_metadata) | ||||||
| { | { | ||||||
|     VERIFY(!value.is_empty()); |     VERIFY(!value.is_special_empty_value()); | ||||||
|     VERIFY(!receiver.is_empty()); |     VERIFY(!receiver.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     // 2. Let ownDesc be ? O.[[GetOwnProperty]](P).
 |     // 2. Let ownDesc be ? O.[[GetOwnProperty]](P).
 | ||||||
|     auto own_descriptor = TRY(internal_get_own_property(property_key)); |     auto own_descriptor = TRY(internal_get_own_property(property_key)); | ||||||
|  | @ -963,8 +963,8 @@ ThrowCompletionOr<bool> Object::internal_set(PropertyKey const& property_key, Va | ||||||
| // 10.1.9.2 OrdinarySetWithOwnDescriptor ( O, P, V, Receiver, ownDesc ), https://tc39.es/ecma262/#sec-ordinarysetwithowndescriptor
 | // 10.1.9.2 OrdinarySetWithOwnDescriptor ( O, P, V, Receiver, ownDesc ), https://tc39.es/ecma262/#sec-ordinarysetwithowndescriptor
 | ||||||
| ThrowCompletionOr<bool> Object::ordinary_set_with_own_descriptor(PropertyKey const& property_key, Value value, Value receiver, Optional<PropertyDescriptor> own_descriptor, CacheablePropertyMetadata* cacheable_metadata) | ThrowCompletionOr<bool> Object::ordinary_set_with_own_descriptor(PropertyKey const& property_key, Value value, Value receiver, Optional<PropertyDescriptor> own_descriptor, CacheablePropertyMetadata* cacheable_metadata) | ||||||
| { | { | ||||||
|     VERIFY(!value.is_empty()); |     VERIFY(!value.is_special_empty_value()); | ||||||
|     VERIFY(!receiver.is_empty()); |     VERIFY(!receiver.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     auto& vm = this->vm(); |     auto& vm = this->vm(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -204,7 +204,6 @@ void Promise::fulfill(Value value) | ||||||
| 
 | 
 | ||||||
|     // 1. Assert: The value of promise.[[PromiseState]] is pending.
 |     // 1. Assert: The value of promise.[[PromiseState]] is pending.
 | ||||||
|     VERIFY(m_state == State::Pending); |     VERIFY(m_state == State::Pending); | ||||||
|     VERIFY(!value.is_empty()); |  | ||||||
| 
 | 
 | ||||||
|     // 2. Let reactions be promise.[[PromiseFulfillReactions]].
 |     // 2. Let reactions be promise.[[PromiseFulfillReactions]].
 | ||||||
|     // NOTE: This is a noop, we do these steps in a slightly different order.
 |     // NOTE: This is a noop, we do these steps in a slightly different order.
 | ||||||
|  | @ -235,7 +234,6 @@ void Promise::reject(Value reason) | ||||||
| 
 | 
 | ||||||
|     // 1. Assert: The value of promise.[[PromiseState]] is pending.
 |     // 1. Assert: The value of promise.[[PromiseState]] is pending.
 | ||||||
|     VERIFY(m_state == State::Pending); |     VERIFY(m_state == State::Pending); | ||||||
|     VERIFY(!reason.is_empty()); |  | ||||||
| 
 | 
 | ||||||
|     // 2. Let reactions be promise.[[PromiseRejectReactions]].
 |     // 2. Let reactions be promise.[[PromiseRejectReactions]].
 | ||||||
|     // NOTE: This is a noop, we do these steps in a slightly different order.
 |     // NOTE: This is a noop, we do these steps in a slightly different order.
 | ||||||
|  |  | ||||||
|  | @ -196,7 +196,7 @@ void PropertyDescriptor::complete() | ||||||
| { | { | ||||||
|     if (is_generic_descriptor() || is_data_descriptor()) { |     if (is_generic_descriptor() || is_data_descriptor()) { | ||||||
|         if (!value.has_value()) |         if (!value.has_value()) | ||||||
|             value = Value {}; |             value = js_undefined(); | ||||||
|         if (!writable.has_value()) |         if (!writable.has_value()) | ||||||
|             writable = false; |             writable = false; | ||||||
|     } else { |     } else { | ||||||
|  |  | ||||||
|  | @ -25,7 +25,7 @@ public: | ||||||
| 
 | 
 | ||||||
|     static ThrowCompletionOr<PropertyKey> from_value(VM& vm, Value value) |     static ThrowCompletionOr<PropertyKey> from_value(VM& vm, Value value) | ||||||
|     { |     { | ||||||
|         VERIFY(!value.is_empty()); |         VERIFY(!value.is_special_empty_value()); | ||||||
|         if (value.is_symbol()) |         if (value.is_symbol()) | ||||||
|             return PropertyKey { value.as_symbol() }; |             return PropertyKey { value.as_symbol() }; | ||||||
|         if (value.is_integral_number() && value.as_double() >= 0 && value.as_double() < NumericLimits<u32>::max()) |         if (value.is_integral_number() && value.as_double() >= 0 && value.as_double() < NumericLimits<u32>::max()) | ||||||
|  |  | ||||||
|  | @ -486,12 +486,10 @@ ThrowCompletionOr<Value> ProxyObject::internal_get(PropertyKey const& property_k | ||||||
| 
 | 
 | ||||||
|     // NOTE: We don't return any cacheable metadata for proxy lookups.
 |     // NOTE: We don't return any cacheable metadata for proxy lookups.
 | ||||||
| 
 | 
 | ||||||
|     VERIFY(!receiver.is_empty()); |     VERIFY(!receiver.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     auto& vm = this->vm(); |     auto& vm = this->vm(); | ||||||
| 
 | 
 | ||||||
|     VERIFY(!receiver.is_empty()); |  | ||||||
| 
 |  | ||||||
|     // 1. Let handler be O.[[ProxyHandler]].
 |     // 1. Let handler be O.[[ProxyHandler]].
 | ||||||
| 
 | 
 | ||||||
|     // 2. If handler is null, throw a TypeError exception.
 |     // 2. If handler is null, throw a TypeError exception.
 | ||||||
|  | @ -559,8 +557,8 @@ ThrowCompletionOr<bool> ProxyObject::internal_set(PropertyKey const& property_ke | ||||||
| 
 | 
 | ||||||
|     auto& vm = this->vm(); |     auto& vm = this->vm(); | ||||||
| 
 | 
 | ||||||
|     VERIFY(!value.is_empty()); |     VERIFY(!value.is_special_empty_value()); | ||||||
|     VERIFY(!receiver.is_empty()); |     VERIFY(!receiver.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     // 1. Let handler be O.[[ProxyHandler]].
 |     // 1. Let handler be O.[[ProxyHandler]].
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -30,7 +30,7 @@ public: | ||||||
|     { |     { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     Reference(Value base, PropertyKey name, Value this_value, bool strict = false) |     Reference(Value base, PropertyKey name, Optional<Value> this_value, bool strict = false) | ||||||
|         : m_base_type(BaseType::Value) |         : m_base_type(BaseType::Value) | ||||||
|         , m_base_value(base) |         , m_base_value(base) | ||||||
|         , m_name(move(name)) |         , m_name(move(name)) | ||||||
|  | @ -90,14 +90,14 @@ public: | ||||||
|     { |     { | ||||||
|         VERIFY(is_property_reference()); |         VERIFY(is_property_reference()); | ||||||
|         if (is_super_reference()) |         if (is_super_reference()) | ||||||
|             return m_this_value; |             return m_this_value.value(); | ||||||
|         return m_base_value; |         return m_base_value; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // 6.2.4.3 IsSuperReference ( V ), https://tc39.es/ecma262/#sec-issuperreference
 |     // 6.2.4.3 IsSuperReference ( V ), https://tc39.es/ecma262/#sec-issuperreference
 | ||||||
|     bool is_super_reference() const |     bool is_super_reference() const | ||||||
|     { |     { | ||||||
|         return !m_this_value.is_empty(); |         return m_this_value.has_value(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // 6.2.4.4 IsPrivateReference ( V ), https://tc39.es/ecma262/#sec-isprivatereference
 |     // 6.2.4.4 IsPrivateReference ( V ), https://tc39.es/ecma262/#sec-isprivatereference
 | ||||||
|  | @ -131,7 +131,7 @@ private: | ||||||
|         mutable Environment* m_base_environment; |         mutable Environment* m_base_environment; | ||||||
|     }; |     }; | ||||||
|     Variant<PropertyKey, PrivateName> m_name; |     Variant<PropertyKey, PrivateName> m_name; | ||||||
|     Value m_this_value; |     Optional<Value> m_this_value; | ||||||
|     bool m_strict { false }; |     bool m_strict { false }; | ||||||
| 
 | 
 | ||||||
|     Optional<EnvironmentCoordinate> m_environment_coordinate; |     Optional<EnvironmentCoordinate> m_environment_coordinate; | ||||||
|  |  | ||||||
|  | @ -158,13 +158,13 @@ ThrowCompletionOr<Value> perform_shadow_realm_eval(VM& vm, StringView source_tex | ||||||
|                 result = result_and_return_register.value.release_error(); |                 result = result_and_return_register.value.release_error(); | ||||||
|             } else { |             } else { | ||||||
|                 // Resulting value is in the accumulator.
 |                 // Resulting value is in the accumulator.
 | ||||||
|                 result = result_and_return_register.return_register_value.value_or(js_undefined()); |                 result = result_and_return_register.return_register_value.is_special_empty_value() ? js_undefined() : result_and_return_register.return_register_value; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // 12. If result.[[Type]] is normal and result.[[Value]] is empty, then
 |     // 12. If result.[[Type]] is normal and result.[[Value]] is empty, then
 | ||||||
|     if (result.type() == Completion::Type::Normal && result.value().is_empty()) { |     if (result.type() == Completion::Type::Normal && result.value().is_special_empty_value()) { | ||||||
|         // a. Set result to NormalCompletion(undefined).
 |         // a. Set result to NormalCompletion(undefined).
 | ||||||
|         result = normal_completion(js_undefined()); |         result = normal_completion(js_undefined()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -161,7 +161,7 @@ inline Value typed_array_get_element(TypedArrayBase const& typed_array, Canonica | ||||||
| template<typename T> | template<typename T> | ||||||
| inline ThrowCompletionOr<void> typed_array_set_element(TypedArrayBase& typed_array, CanonicalIndex property_index, Value value) | inline ThrowCompletionOr<void> typed_array_set_element(TypedArrayBase& typed_array, CanonicalIndex property_index, Value value) | ||||||
| { | { | ||||||
|     VERIFY(!value.is_empty()); |     VERIFY(!value.is_special_empty_value()); | ||||||
|     auto& vm = typed_array.vm(); |     auto& vm = typed_array.vm(); | ||||||
| 
 | 
 | ||||||
|     Value num_value; |     Value num_value; | ||||||
|  | @ -329,7 +329,7 @@ public: | ||||||
|     // 10.4.5.5 [[Get]] ( P, Receiver ), https://tc39.es/ecma262/#sec-typedarray-get
 |     // 10.4.5.5 [[Get]] ( P, Receiver ), https://tc39.es/ecma262/#sec-typedarray-get
 | ||||||
|     virtual ThrowCompletionOr<Value> internal_get(PropertyKey const& property_key, Value receiver, CacheablePropertyMetadata* cacheable_metadata, PropertyLookupPhase phase) const override |     virtual ThrowCompletionOr<Value> internal_get(PropertyKey const& property_key, Value receiver, CacheablePropertyMetadata* cacheable_metadata, PropertyLookupPhase phase) const override | ||||||
|     { |     { | ||||||
|         VERIFY(!receiver.is_empty()); |         VERIFY(!receiver.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|         // NOTE: If the property name is a number type (An implementation-defined optimized
 |         // NOTE: If the property name is a number type (An implementation-defined optimized
 | ||||||
|         // property key type), it can be treated as a string property that will transparently be
 |         // property key type), it can be treated as a string property that will transparently be
 | ||||||
|  | @ -354,8 +354,8 @@ public: | ||||||
|     // 10.4.5.6 [[Set]] ( P, V, Receiver ), https://tc39.es/ecma262/#sec-integer-indexed-exotic-objects-set-p-v-receiver
 |     // 10.4.5.6 [[Set]] ( P, V, Receiver ), https://tc39.es/ecma262/#sec-integer-indexed-exotic-objects-set-p-v-receiver
 | ||||||
|     virtual ThrowCompletionOr<bool> internal_set(PropertyKey const& property_key, Value value, Value receiver, CacheablePropertyMetadata*) override |     virtual ThrowCompletionOr<bool> internal_set(PropertyKey const& property_key, Value value, Value receiver, CacheablePropertyMetadata*) override | ||||||
|     { |     { | ||||||
|         VERIFY(!value.is_empty()); |         VERIFY(!value.is_special_empty_value()); | ||||||
|         VERIFY(!receiver.is_empty()); |         VERIFY(!receiver.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|         // NOTE: If the property name is a number type (An implementation-defined optimized
 |         // NOTE: If the property name is a number type (An implementation-defined optimized
 | ||||||
|         // property key type), it can be treated as a string property that will transparently be
 |         // property key type), it can be treated as a string property that will transparently be
 | ||||||
|  |  | ||||||
|  | @ -175,7 +175,7 @@ public: | ||||||
| 
 | 
 | ||||||
|     Value this_value() const |     Value this_value() const | ||||||
|     { |     { | ||||||
|         return running_execution_context().this_value; |         return running_execution_context().this_value.value(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     ThrowCompletionOr<Value> resolve_this_binding(); |     ThrowCompletionOr<Value> resolve_this_binding(); | ||||||
|  |  | ||||||
|  | @ -559,7 +559,7 @@ ThrowCompletionOr<Value> Value::to_primitive_slow_case(VM& vm, PreferredType pre | ||||||
| ThrowCompletionOr<GC::Ref<Object>> Value::to_object(VM& vm) const | ThrowCompletionOr<GC::Ref<Object>> Value::to_object(VM& vm) const | ||||||
| { | { | ||||||
|     auto& realm = *vm.current_realm(); |     auto& realm = *vm.current_realm(); | ||||||
|     VERIFY(!is_empty()); |     VERIFY(!is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     // Number
 |     // Number
 | ||||||
|     if (is_number()) { |     if (is_number()) { | ||||||
|  | @ -712,7 +712,7 @@ double string_to_number(StringView string) | ||||||
| // 7.1.4 ToNumber ( argument ), https://tc39.es/ecma262/#sec-tonumber
 | // 7.1.4 ToNumber ( argument ), https://tc39.es/ecma262/#sec-tonumber
 | ||||||
| ThrowCompletionOr<Value> Value::to_number_slow_case(VM& vm) const | ThrowCompletionOr<Value> Value::to_number_slow_case(VM& vm) const | ||||||
| { | { | ||||||
|     VERIFY(!is_empty()); |     VERIFY(!is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|     // 1. If argument is a Number, return argument.
 |     // 1. If argument is a Number, return argument.
 | ||||||
|     if (is_number()) |     if (is_number()) | ||||||
|  |  | ||||||
|  | @ -95,7 +95,7 @@ public: | ||||||
| 
 | 
 | ||||||
|     [[nodiscard]] u16 tag() const { return m_value.tag; } |     [[nodiscard]] u16 tag() const { return m_value.tag; } | ||||||
| 
 | 
 | ||||||
|     bool is_empty() const { return m_value.tag == EMPTY_TAG; } |     bool is_special_empty_value() const { return m_value.tag == EMPTY_TAG; } | ||||||
|     bool is_undefined() const { return m_value.tag == UNDEFINED_TAG; } |     bool is_undefined() const { return m_value.tag == UNDEFINED_TAG; } | ||||||
|     bool is_null() const { return m_value.tag == NULL_TAG; } |     bool is_null() const { return m_value.tag == NULL_TAG; } | ||||||
|     bool is_number() const { return is_double() || is_int32(); } |     bool is_number() const { return is_double() || is_int32(); } | ||||||
|  | @ -155,7 +155,7 @@ public: | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     Value() |     Value() | ||||||
|         : Value(EMPTY_TAG << GC::TAG_SHIFT, (u64)0) |         : Value(UNDEFINED_TAG << GC::TAG_SHIFT, (u64)0) | ||||||
|     { |     { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -379,12 +379,14 @@ public: | ||||||
| 
 | 
 | ||||||
|     [[nodiscard]] String to_string_without_side_effects() const; |     [[nodiscard]] String to_string_without_side_effects() const; | ||||||
| 
 | 
 | ||||||
|  | #if 0 | ||||||
|     Value value_or(Value fallback) const |     Value value_or(Value fallback) const | ||||||
|     { |     { | ||||||
|         if (is_empty()) |         if (is_special_empty_value()) | ||||||
|             return fallback; |             return fallback; | ||||||
|         return *this; |         return *this; | ||||||
|     } |     } | ||||||
|  | #endif | ||||||
| 
 | 
 | ||||||
|     [[nodiscard]] GC::Ref<PrimitiveString> typeof_(VM&) const; |     [[nodiscard]] GC::Ref<PrimitiveString> typeof_(VM&) const; | ||||||
| 
 | 
 | ||||||
|  | @ -424,6 +426,13 @@ private: | ||||||
|     ThrowCompletionOr<Value> to_numeric_slow_case(VM&) const; |     ThrowCompletionOr<Value> to_numeric_slow_case(VM&) const; | ||||||
|     ThrowCompletionOr<Value> to_primitive_slow_case(VM&, PreferredType) const; |     ThrowCompletionOr<Value> to_primitive_slow_case(VM&, PreferredType) const; | ||||||
| 
 | 
 | ||||||
|  |     enum class EmptyTag { Empty }; | ||||||
|  | 
 | ||||||
|  |     Value(EmptyTag) | ||||||
|  |         : Value(EMPTY_TAG << GC::TAG_SHIFT, (u64)0) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     Value(u64 tag, u64 val) |     Value(u64 tag, u64 val) | ||||||
|     { |     { | ||||||
|         ASSERT(!(tag & val)); |         ASSERT(!(tag & val)); | ||||||
|  | @ -460,6 +469,7 @@ private: | ||||||
| 
 | 
 | ||||||
|     friend Value js_undefined(); |     friend Value js_undefined(); | ||||||
|     friend Value js_null(); |     friend Value js_null(); | ||||||
|  |     friend Value js_special_empty_value(); | ||||||
|     friend ThrowCompletionOr<Value> greater_than(VM&, Value lhs, Value rhs); |     friend ThrowCompletionOr<Value> greater_than(VM&, Value lhs, Value rhs); | ||||||
|     friend ThrowCompletionOr<Value> greater_than_equals(VM&, Value lhs, Value rhs); |     friend ThrowCompletionOr<Value> greater_than_equals(VM&, Value lhs, Value rhs); | ||||||
|     friend ThrowCompletionOr<Value> less_than(VM&, Value lhs, Value rhs); |     friend ThrowCompletionOr<Value> less_than(VM&, Value lhs, Value rhs); | ||||||
|  | @ -478,6 +488,11 @@ inline Value js_null() | ||||||
|     return Value(NULL_TAG << GC::TAG_SHIFT, (u64)0); |     return Value(NULL_TAG << GC::TAG_SHIFT, (u64)0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | inline Value js_special_empty_value() | ||||||
|  | { | ||||||
|  |     return Value(Value::EmptyTag::Empty); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| inline Value js_nan() | inline Value js_nan() | ||||||
| { | { | ||||||
|     return Value(NAN); |     return Value(NAN); | ||||||
|  | @ -600,12 +615,12 @@ public: | ||||||
| 
 | 
 | ||||||
|     void clear() |     void clear() | ||||||
|     { |     { | ||||||
|         m_value = {}; |         m_value = JS::js_special_empty_value(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [[nodiscard]] bool has_value() const |     [[nodiscard]] bool has_value() const | ||||||
|     { |     { | ||||||
|         return !m_value.is_empty(); |         return !m_value.is_special_empty_value(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [[nodiscard]] JS::Value& value() & |     [[nodiscard]] JS::Value& value() & | ||||||
|  | @ -634,7 +649,7 @@ public: | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|     JS::Value m_value; |     JS::Value m_value { JS::js_special_empty_value() }; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
|  | @ -690,7 +705,7 @@ template<> | ||||||
| struct Formatter<JS::Value> : Formatter<StringView> { | struct Formatter<JS::Value> : Formatter<StringView> { | ||||||
|     ErrorOr<void> format(FormatBuilder& builder, JS::Value value) |     ErrorOr<void> format(FormatBuilder& builder, JS::Value value) | ||||||
|     { |     { | ||||||
|         if (value.is_empty()) |         if (value.is_special_empty_value()) | ||||||
|             return Formatter<StringView>::format(builder, "<empty>"sv); |             return Formatter<StringView>::format(builder, "<empty>"sv); | ||||||
|         return Formatter<StringView>::format(builder, value.to_string_without_side_effects()); |         return Formatter<StringView>::format(builder, value.to_string_without_side_effects()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -16,7 +16,7 @@ namespace JS { | ||||||
| struct ValueTraits : public Traits<Value> { | struct ValueTraits : public Traits<Value> { | ||||||
|     static unsigned hash(Value value) |     static unsigned hash(Value value) | ||||||
|     { |     { | ||||||
|         VERIFY(!value.is_empty()); |         VERIFY(!value.is_special_empty_value()); | ||||||
|         if (value.is_string()) |         if (value.is_string()) | ||||||
|             return value.as_string().utf8_string().hash(); |             return value.as_string().utf8_string().hash(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -727,7 +727,7 @@ ThrowCompletionOr<void> SourceTextModule::execute_module(VM& vm, GC::Ptr<Promise | ||||||
|                 result = result_and_return_register.value.release_error(); |                 result = result_and_return_register.value.release_error(); | ||||||
|             } else { |             } else { | ||||||
|                 // Resulting value is in the accumulator.
 |                 // Resulting value is in the accumulator.
 | ||||||
|                 result = result_and_return_register.return_register_value.value_or(js_undefined()); |                 result = result_and_return_register.return_register_value.is_special_empty_value() ? js_undefined() : result_and_return_register.return_register_value; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -436,8 +436,8 @@ inline JSFileResult TestRunner::run_file_test(ByteString const& test_path) | ||||||
|             StringBuilder detail_builder; |             StringBuilder detail_builder; | ||||||
| 
 | 
 | ||||||
|             auto& error_object = error.as_object(); |             auto& error_object = error.as_object(); | ||||||
|             auto name = error_object.get_without_side_effects(g_vm->names.name).value_or(JS::js_undefined()); |             auto name = error_object.get_without_side_effects(g_vm->names.name); | ||||||
|             auto message = error_object.get_without_side_effects(g_vm->names.message).value_or(JS::js_undefined()); |             auto message = error_object.get_without_side_effects(g_vm->names.message); | ||||||
| 
 | 
 | ||||||
|             if (name.is_accessor() || message.is_accessor()) { |             if (name.is_accessor() || message.is_accessor()) { | ||||||
|                 detail_builder.append(error.to_string_without_side_effects()); |                 detail_builder.append(error.to_string_without_side_effects()); | ||||||
|  |  | ||||||
|  | @ -22,8 +22,8 @@ void report_exception_to_console(JS::Value value, JS::Realm& realm, ErrorInPromi | ||||||
|     if (value.is_object()) { |     if (value.is_object()) { | ||||||
|         auto& object = value.as_object(); |         auto& object = value.as_object(); | ||||||
|         auto& vm = object.vm(); |         auto& vm = object.vm(); | ||||||
|         auto name = object.get_without_side_effects(vm.names.name).value_or(JS::js_undefined()); |         auto name = object.get_without_side_effects(vm.names.name); | ||||||
|         auto message = object.get_without_side_effects(vm.names.message).value_or(JS::js_undefined()); |         auto message = object.get_without_side_effects(vm.names.message); | ||||||
|         if (name.is_accessor() || message.is_accessor()) { |         if (name.is_accessor() || message.is_accessor()) { | ||||||
|             // The result is not going to be useful, let's just print the value. This affects DOMExceptions, for example.
 |             // The result is not going to be useful, let's just print the value. This affects DOMExceptions, for example.
 | ||||||
|             if (is<WebIDL::DOMException>(object)) { |             if (is<WebIDL::DOMException>(object)) { | ||||||
|  |  | ||||||
|  | @ -51,14 +51,14 @@ static JS::Value create_close_sentinel() | ||||||
| { | { | ||||||
|     // The close sentinel is a unique value enqueued into [[queue]], in lieu of a chunk, to signal that the stream is closed. It is only used internally, and is never exposed to web developers.
 |     // The close sentinel is a unique value enqueued into [[queue]], in lieu of a chunk, to signal that the stream is closed. It is only used internally, and is never exposed to web developers.
 | ||||||
|     // Note: We use the empty Value to signal this as, similarly to the note above, the empty value is not exposed to nor creatable by web developers.
 |     // Note: We use the empty Value to signal this as, similarly to the note above, the empty value is not exposed to nor creatable by web developers.
 | ||||||
|     return {}; |     return JS::js_special_empty_value(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // https://streams.spec.whatwg.org/#close-sentinel
 | // https://streams.spec.whatwg.org/#close-sentinel
 | ||||||
| // Non-standard function that implements the "If value is a close sentinel" check.
 | // Non-standard function that implements the "If value is a close sentinel" check.
 | ||||||
| static bool is_close_sentinel(JS::Value value) | static bool is_close_sentinel(JS::Value value) | ||||||
| { | { | ||||||
|     return value.is_empty(); |     return value.is_special_empty_value(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // NON-STANDARD: Can be used instead of CreateReadableStream in cases where we need to set up a newly allocated
 | // NON-STANDARD: Can be used instead of CreateReadableStream in cases where we need to set up a newly allocated
 | ||||||
|  |  | ||||||
|  | @ -103,8 +103,8 @@ void DevToolsConsoleClient::report_exception(JS::Error const& exception, bool in | ||||||
| { | { | ||||||
|     auto& vm = exception.vm(); |     auto& vm = exception.vm(); | ||||||
| 
 | 
 | ||||||
|     auto name = exception.get_without_side_effects(vm.names.name).value_or(JS::js_undefined()); |     auto name = exception.get_without_side_effects(vm.names.name); | ||||||
|     auto message = exception.get_without_side_effects(vm.names.message).value_or(JS::js_undefined()); |     auto message = exception.get_without_side_effects(vm.names.message); | ||||||
| 
 | 
 | ||||||
|     Vector<WebView::StackFrame> trace; |     Vector<WebView::StackFrame> trace; | ||||||
|     trace.ensure_capacity(exception.traceback().size()); |     trace.ensure_capacity(exception.traceback().size()); | ||||||
|  |  | ||||||
|  | @ -86,7 +86,7 @@ TEST_CASE(non_canon_nans) | ||||||
|         EXPECT(!val.is_integral_number());     \ |         EXPECT(!val.is_integral_number());     \ | ||||||
|         EXPECT(!val.is_finite_number());       \ |         EXPECT(!val.is_finite_number());       \ | ||||||
|         EXPECT(!val.is_infinity());            \ |         EXPECT(!val.is_infinity());            \ | ||||||
|         EXPECT(!val.is_empty());               \ |         EXPECT(!val.is_special_empty_value()); \ | ||||||
|         EXPECT(!val.is_nullish());             \ |         EXPECT(!val.is_nullish());             \ | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -91,7 +91,7 @@ static ErrorOr<void, TestError> run_program(InterpreterT& interpreter, ScriptOrM | ||||||
|             auto& object = error_value.as_object(); |             auto& object = error_value.as_object(); | ||||||
| 
 | 
 | ||||||
|             auto name = object.get_without_side_effects("name"_fly_string); |             auto name = object.get_without_side_effects("name"_fly_string); | ||||||
|             if (!name.is_empty() && !name.is_accessor()) { |             if (!name.is_undefined() && !name.is_accessor()) { | ||||||
|                 error.type = name.to_string_without_side_effects(); |                 error.type = name.to_string_without_side_effects(); | ||||||
|             } else { |             } else { | ||||||
|                 auto constructor = object.get_without_side_effects("constructor"_fly_string); |                 auto constructor = object.get_without_side_effects("constructor"_fly_string); | ||||||
|  | @ -103,7 +103,7 @@ static ErrorOr<void, TestError> run_program(InterpreterT& interpreter, ScriptOrM | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             auto message = object.get_without_side_effects("message"_fly_string); |             auto message = object.get_without_side_effects("message"_fly_string); | ||||||
|             if (!message.is_empty() && !message.is_accessor()) |             if (!message.is_undefined() && !message.is_accessor()) | ||||||
|                 error.details = message.to_string_without_side_effects(); |                 error.details = message.to_string_without_side_effects(); | ||||||
|         } |         } | ||||||
|         if (error.type.is_empty()) |         if (error.type.is_empty()) | ||||||
|  |  | ||||||
|  | @ -778,7 +778,7 @@ ErrorOr<int> serenity_main(Main::Arguments arguments) | ||||||
|                 if (value_or_error.is_error()) |                 if (value_or_error.is_error()) | ||||||
|                     return {}; |                     return {}; | ||||||
|                 auto variable = value_or_error.value(); |                 auto variable = value_or_error.value(); | ||||||
|                 VERIFY(!variable.is_empty()); |                 VERIFY(!variable.is_special_empty_value()); | ||||||
| 
 | 
 | ||||||
|                 if (!variable.is_object()) |                 if (!variable.is_object()) | ||||||
|                     break; |                     break; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Andreas Kling
						Andreas Kling