summaryrefslogtreecommitdiff
path: root/Source/JavaScriptCore/bytecode/BytecodeUseDef.h
diff options
context:
space:
mode:
authorKonstantin Tokarev <annulen@yandex.ru>2016-08-25 19:20:41 +0300
committerKonstantin Tokarev <annulen@yandex.ru>2017-02-02 12:30:55 +0000
commit6882a04fb36642862b11efe514251d32070c3d65 (patch)
treeb7959826000b061fd5ccc7512035c7478742f7b0 /Source/JavaScriptCore/bytecode/BytecodeUseDef.h
parentab6df191029eeeb0b0f16f127d553265659f739e (diff)
downloadqtwebkit-6882a04fb36642862b11efe514251d32070c3d65.tar.gz
Imported QtWebKit TP3 (git b57bc6801f1876c3220d5a4bfea33d620d477443)
Change-Id: I3b1d8a2808782c9f34d50240000e20cb38d3680f Reviewed-by: Konstantin Tokarev <annulen@yandex.ru>
Diffstat (limited to 'Source/JavaScriptCore/bytecode/BytecodeUseDef.h')
-rw-r--r--Source/JavaScriptCore/bytecode/BytecodeUseDef.h447
1 files changed, 447 insertions, 0 deletions
diff --git a/Source/JavaScriptCore/bytecode/BytecodeUseDef.h b/Source/JavaScriptCore/bytecode/BytecodeUseDef.h
new file mode 100644
index 000000000..14a69f68a
--- /dev/null
+++ b/Source/JavaScriptCore/bytecode/BytecodeUseDef.h
@@ -0,0 +1,447 @@
+/*
+ * Copyright (C) 2013, 2015 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
+ * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
+ * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
+ * THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#ifndef BytecodeUseDef_h
+#define BytecodeUseDef_h
+
+#include "CodeBlock.h"
+
+namespace JSC {
+
+template<typename Functor>
+void computeUsesForBytecodeOffset(
+ CodeBlock* codeBlock, BytecodeBasicBlock* block, unsigned bytecodeOffset, const Functor& functor)
+{
+ Interpreter* interpreter = codeBlock->vm()->interpreter;
+ Instruction* instructionsBegin = codeBlock->instructions().begin();
+ Instruction* instruction = &instructionsBegin[bytecodeOffset];
+ OpcodeID opcodeID = interpreter->getOpcodeID(instruction->u.opcode);
+ switch (opcodeID) {
+ // No uses.
+ case op_new_regexp:
+ case op_new_array_buffer:
+ case op_throw_static_error:
+ case op_debug:
+ case op_jneq_ptr:
+ case op_loop_hint:
+ case op_jmp:
+ case op_new_object:
+ case op_enter:
+ case op_catch:
+ case op_profile_control_flow:
+ case op_create_direct_arguments:
+ case op_create_out_of_band_arguments:
+ case op_get_rest_length:
+ case op_watchdog:
+ return;
+ case op_assert:
+ case op_get_scope:
+ case op_to_this:
+ case op_check_tdz:
+ case op_profile_will_call:
+ case op_profile_did_call:
+ case op_profile_type:
+ case op_throw:
+ case op_end:
+ case op_ret:
+ case op_jtrue:
+ case op_jfalse:
+ case op_jeq_null:
+ case op_jneq_null:
+ case op_dec:
+ case op_inc:
+ case op_resume: {
+ ASSERT(opcodeLengths[opcodeID] > 1);
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ return;
+ }
+ case op_jlesseq:
+ case op_jgreater:
+ case op_jgreatereq:
+ case op_jnless:
+ case op_jnlesseq:
+ case op_jngreater:
+ case op_jngreatereq:
+ case op_jless:
+ case op_copy_rest: {
+ ASSERT(opcodeLengths[opcodeID] > 2);
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[2].u.operand);
+ return;
+ }
+ case op_put_by_val_direct:
+ case op_put_by_val: {
+ ASSERT(opcodeLengths[opcodeID] > 3);
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[2].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[3].u.operand);
+ return;
+ }
+ case op_put_by_index:
+ case op_put_by_id:
+ case op_put_to_scope:
+ case op_put_to_arguments: {
+ ASSERT(opcodeLengths[opcodeID] > 3);
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[3].u.operand);
+ return;
+ }
+ case op_put_getter_by_id:
+ case op_put_setter_by_id: {
+ ASSERT(opcodeLengths[opcodeID] > 4);
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[4].u.operand);
+ return;
+ }
+ case op_put_getter_setter_by_id: {
+ ASSERT(opcodeLengths[opcodeID] > 5);
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[4].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[5].u.operand);
+ return;
+ }
+ case op_put_getter_by_val:
+ case op_put_setter_by_val: {
+ ASSERT(opcodeLengths[opcodeID] > 4);
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[2].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[4].u.operand);
+ return;
+ }
+ case op_get_property_enumerator:
+ case op_get_enumerable_length:
+ case op_new_func_exp:
+ case op_new_generator_func_exp:
+ case op_new_arrow_func_exp:
+ case op_to_index_string:
+ case op_create_lexical_environment:
+ case op_resolve_scope:
+ case op_get_from_scope:
+ case op_to_primitive:
+ case op_get_by_id:
+ case op_get_array_length:
+ case op_typeof:
+ case op_is_undefined:
+ case op_is_boolean:
+ case op_is_number:
+ case op_is_string:
+ case op_is_object:
+ case op_is_object_or_null:
+ case op_is_function:
+ case op_to_number:
+ case op_to_string:
+ case op_negate:
+ case op_neq_null:
+ case op_eq_null:
+ case op_not:
+ case op_mov:
+ case op_new_array_with_size:
+ case op_create_this:
+ case op_del_by_id:
+ case op_unsigned:
+ case op_new_func:
+ case op_new_generator_func:
+ case op_get_parent_scope:
+ case op_create_scoped_arguments:
+ case op_get_from_arguments: {
+ ASSERT(opcodeLengths[opcodeID] > 2);
+ functor(codeBlock, instruction, opcodeID, instruction[2].u.operand);
+ return;
+ }
+ case op_has_generic_property:
+ case op_has_indexed_property:
+ case op_enumerator_structure_pname:
+ case op_enumerator_generic_pname:
+ case op_get_by_val:
+ case op_in:
+ case op_overrides_has_instance:
+ case op_instanceof:
+ case op_add:
+ case op_mul:
+ case op_div:
+ case op_mod:
+ case op_sub:
+ case op_lshift:
+ case op_rshift:
+ case op_urshift:
+ case op_bitand:
+ case op_bitxor:
+ case op_bitor:
+ case op_less:
+ case op_lesseq:
+ case op_greater:
+ case op_greatereq:
+ case op_nstricteq:
+ case op_stricteq:
+ case op_neq:
+ case op_eq:
+ case op_push_with_scope:
+ case op_del_by_val: {
+ ASSERT(opcodeLengths[opcodeID] > 3);
+ functor(codeBlock, instruction, opcodeID, instruction[2].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[3].u.operand);
+ return;
+ }
+ case op_instanceof_custom:
+ case op_has_structure_property:
+ case op_construct_varargs:
+ case op_call_varargs:
+ case op_tail_call_varargs: {
+ ASSERT(opcodeLengths[opcodeID] > 4);
+ functor(codeBlock, instruction, opcodeID, instruction[2].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[3].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[4].u.operand);
+ return;
+ }
+ case op_get_direct_pname: {
+ ASSERT(opcodeLengths[opcodeID] > 5);
+ functor(codeBlock, instruction, opcodeID, instruction[2].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[3].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[4].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[5].u.operand);
+ return;
+ }
+ case op_switch_string:
+ case op_switch_char:
+ case op_switch_imm: {
+ ASSERT(opcodeLengths[opcodeID] > 3);
+ functor(codeBlock, instruction, opcodeID, instruction[3].u.operand);
+ return;
+ }
+ case op_new_array:
+ case op_strcat: {
+ int base = instruction[2].u.operand;
+ int count = instruction[3].u.operand;
+ for (int i = 0; i < count; i++)
+ functor(codeBlock, instruction, opcodeID, base - i);
+ return;
+ }
+ case op_construct:
+ case op_call_eval:
+ case op_call:
+ case op_tail_call: {
+ functor(codeBlock, instruction, opcodeID, instruction[2].u.operand);
+ int argCount = instruction[3].u.operand;
+ int registerOffset = -instruction[4].u.operand;
+ int lastArg = registerOffset + CallFrame::thisArgumentOffset();
+ for (int i = 0; i < argCount; i++)
+ functor(codeBlock, instruction, opcodeID, lastArg + i);
+ return;
+ }
+ case op_save: {
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ unsigned mergePointBytecodeOffset = bytecodeOffset + instruction[3].u.operand;
+ BytecodeBasicBlock* mergePointBlock = nullptr;
+ for (BytecodeBasicBlock* successor : block->successors()) {
+ if (successor->leaderBytecodeOffset() == mergePointBytecodeOffset) {
+ mergePointBlock = successor;
+ break;
+ }
+ }
+ ASSERT(mergePointBlock);
+ mergePointBlock->in().forEachSetBit([&](unsigned local) {
+ functor(codeBlock, instruction, opcodeID, virtualRegisterForLocal(local).offset());
+ });
+ return;
+ }
+ default:
+ RELEASE_ASSERT_NOT_REACHED();
+ break;
+ }
+}
+
+template<typename Functor>
+void computeDefsForBytecodeOffset(CodeBlock* codeBlock, BytecodeBasicBlock* block, unsigned bytecodeOffset, const Functor& functor)
+{
+ Interpreter* interpreter = codeBlock->vm()->interpreter;
+ Instruction* instructionsBegin = codeBlock->instructions().begin();
+ Instruction* instruction = &instructionsBegin[bytecodeOffset];
+ OpcodeID opcodeID = interpreter->getOpcodeID(instruction->u.opcode);
+ switch (opcodeID) {
+ // These don't define anything.
+ case op_copy_rest:
+ case op_put_to_scope:
+ case op_end:
+ case op_profile_will_call:
+ case op_profile_did_call:
+ case op_throw:
+ case op_throw_static_error:
+ case op_save:
+ case op_assert:
+ case op_debug:
+ case op_ret:
+ case op_jmp:
+ case op_jtrue:
+ case op_jfalse:
+ case op_jeq_null:
+ case op_jneq_null:
+ case op_jneq_ptr:
+ case op_jless:
+ case op_jlesseq:
+ case op_jgreater:
+ case op_jgreatereq:
+ case op_jnless:
+ case op_jnlesseq:
+ case op_jngreater:
+ case op_jngreatereq:
+ case op_loop_hint:
+ case op_switch_imm:
+ case op_switch_char:
+ case op_switch_string:
+ case op_put_by_id:
+ case op_put_getter_by_id:
+ case op_put_setter_by_id:
+ case op_put_getter_setter_by_id:
+ case op_put_getter_by_val:
+ case op_put_setter_by_val:
+ case op_put_by_val:
+ case op_put_by_val_direct:
+ case op_put_by_index:
+ case op_profile_type:
+ case op_profile_control_flow:
+ case op_put_to_arguments:
+ case op_watchdog:
+#define LLINT_HELPER_OPCODES(opcode, length) case opcode:
+ FOR_EACH_LLINT_OPCODE_EXTENSION(LLINT_HELPER_OPCODES);
+#undef LLINT_HELPER_OPCODES
+ return;
+ // These all have a single destination for the first argument.
+ case op_to_index_string:
+ case op_get_enumerable_length:
+ case op_has_indexed_property:
+ case op_has_structure_property:
+ case op_has_generic_property:
+ case op_get_direct_pname:
+ case op_get_property_enumerator:
+ case op_enumerator_structure_pname:
+ case op_enumerator_generic_pname:
+ case op_get_parent_scope:
+ case op_push_with_scope:
+ case op_create_lexical_environment:
+ case op_resolve_scope:
+ case op_strcat:
+ case op_to_primitive:
+ case op_create_this:
+ case op_new_array:
+ case op_new_array_buffer:
+ case op_new_array_with_size:
+ case op_new_regexp:
+ case op_new_func:
+ case op_new_func_exp:
+ case op_new_generator_func:
+ case op_new_generator_func_exp:
+ case op_new_arrow_func_exp:
+ case op_call_varargs:
+ case op_tail_call_varargs:
+ case op_construct_varargs:
+ case op_get_from_scope:
+ case op_call:
+ case op_tail_call:
+ case op_call_eval:
+ case op_construct:
+ case op_get_by_id:
+ case op_get_array_length:
+ case op_overrides_has_instance:
+ case op_instanceof:
+ case op_instanceof_custom:
+ case op_get_by_val:
+ case op_typeof:
+ case op_is_undefined:
+ case op_is_boolean:
+ case op_is_number:
+ case op_is_string:
+ case op_is_object:
+ case op_is_object_or_null:
+ case op_is_function:
+ case op_in:
+ case op_to_number:
+ case op_to_string:
+ case op_negate:
+ case op_add:
+ case op_mul:
+ case op_div:
+ case op_mod:
+ case op_sub:
+ case op_lshift:
+ case op_rshift:
+ case op_urshift:
+ case op_bitand:
+ case op_bitxor:
+ case op_bitor:
+ case op_inc:
+ case op_dec:
+ case op_eq:
+ case op_neq:
+ case op_stricteq:
+ case op_nstricteq:
+ case op_less:
+ case op_lesseq:
+ case op_greater:
+ case op_greatereq:
+ case op_neq_null:
+ case op_eq_null:
+ case op_not:
+ case op_mov:
+ case op_new_object:
+ case op_to_this:
+ case op_check_tdz:
+ case op_get_scope:
+ case op_create_direct_arguments:
+ case op_create_scoped_arguments:
+ case op_create_out_of_band_arguments:
+ case op_del_by_id:
+ case op_del_by_val:
+ case op_unsigned:
+ case op_get_from_arguments:
+ case op_get_rest_length: {
+ ASSERT(opcodeLengths[opcodeID] > 1);
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ return;
+ }
+ case op_catch: {
+ ASSERT(opcodeLengths[opcodeID] > 2);
+ functor(codeBlock, instruction, opcodeID, instruction[1].u.operand);
+ functor(codeBlock, instruction, opcodeID, instruction[2].u.operand);
+ return;
+ }
+ case op_enter: {
+ for (unsigned i = codeBlock->m_numVars; i--;)
+ functor(codeBlock, instruction, opcodeID, virtualRegisterForLocal(i).offset());
+ return;
+ }
+ case op_resume: {
+ RELEASE_ASSERT(block->successors().size() == 1);
+ block->successors()[0]->in().forEachSetBit([&](unsigned local) {
+ functor(codeBlock, instruction, opcodeID, virtualRegisterForLocal(local).offset());
+ });
+ return;
+ }
+ }
+}
+
+} // namespace JSC
+
+#endif // BytecodeUseDef_h
+