mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 09:42:35 +00:00 
			
		
		
		
	 482a827346
			
		
	
	
		482a827346
		
	
	
	
	
		
			
			Previously it would throw instead of returning "undefined" for `typeof Identifier` if Identifier does not exist.
		
			
				
	
	
		
			152 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/Forward.h>
 | |
| #include <AK/Span.h>
 | |
| #include <LibJS/Forward.h>
 | |
| 
 | |
| #define ENUMERATE_BYTECODE_OPS(O)    \
 | |
|     O(Add)                           \
 | |
|     O(BitwiseAnd)                    \
 | |
|     O(BitwiseNot)                    \
 | |
|     O(BitwiseOr)                     \
 | |
|     O(BitwiseXor)                    \
 | |
|     O(Call)                          \
 | |
|     O(ConcatString)                  \
 | |
|     O(ContinuePendingUnwind)         \
 | |
|     O(CopyObjectExcludingProperties) \
 | |
|     O(CreateEnvironment)             \
 | |
|     O(CreateVariable)                \
 | |
|     O(Decrement)                     \
 | |
|     O(DeleteById)                    \
 | |
|     O(DeleteByValue)                 \
 | |
|     O(DeleteVariable)                \
 | |
|     O(Div)                           \
 | |
|     O(EnterUnwindContext)            \
 | |
|     O(EnterObjectEnvironment)        \
 | |
|     O(Exp)                           \
 | |
|     O(FinishUnwind)                  \
 | |
|     O(GetById)                       \
 | |
|     O(GetByValue)                    \
 | |
|     O(GetIterator)                   \
 | |
|     O(GetNewTarget)                  \
 | |
|     O(GetObjectPropertyIterator)     \
 | |
|     O(GetVariable)                   \
 | |
|     O(GreaterThan)                   \
 | |
|     O(GreaterThanEquals)             \
 | |
|     O(In)                            \
 | |
|     O(Increment)                     \
 | |
|     O(InstanceOf)                    \
 | |
|     O(IteratorNext)                  \
 | |
|     O(IteratorResultDone)            \
 | |
|     O(IteratorResultValue)           \
 | |
|     O(IteratorToArray)               \
 | |
|     O(Jump)                          \
 | |
|     O(JumpConditional)               \
 | |
|     O(JumpNullish)                   \
 | |
|     O(JumpUndefined)                 \
 | |
|     O(LeaveEnvironment)              \
 | |
|     O(LeaveUnwindContext)            \
 | |
|     O(LeftShift)                     \
 | |
|     O(LessThan)                      \
 | |
|     O(LessThanEquals)                \
 | |
|     O(Load)                          \
 | |
|     O(LoadImmediate)                 \
 | |
|     O(LooselyEquals)                 \
 | |
|     O(LooselyInequals)               \
 | |
|     O(Mod)                           \
 | |
|     O(Mul)                           \
 | |
|     O(NewArray)                      \
 | |
|     O(NewBigInt)                     \
 | |
|     O(NewClass)                      \
 | |
|     O(NewFunction)                   \
 | |
|     O(NewObject)                     \
 | |
|     O(NewRegExp)                     \
 | |
|     O(NewString)                     \
 | |
|     O(Not)                           \
 | |
|     O(PushDeclarativeEnvironment)    \
 | |
|     O(PutById)                       \
 | |
|     O(PutByValue)                    \
 | |
|     O(ResolveThisBinding)            \
 | |
|     O(Return)                        \
 | |
|     O(RightShift)                    \
 | |
|     O(SetVariable)                   \
 | |
|     O(Store)                         \
 | |
|     O(StrictlyEquals)                \
 | |
|     O(StrictlyInequals)              \
 | |
|     O(Sub)                           \
 | |
|     O(Throw)                         \
 | |
|     O(Typeof)                        \
 | |
|     O(TypeofVariable)                \
 | |
|     O(UnaryMinus)                    \
 | |
|     O(UnaryPlus)                     \
 | |
|     O(UnsignedRightShift)            \
 | |
|     O(Yield)
 | |
| 
 | |
| namespace JS::Bytecode {
 | |
| 
 | |
| class Instruction {
 | |
| public:
 | |
|     constexpr static bool IsTerminator = false;
 | |
| 
 | |
|     enum class Type {
 | |
| #define __BYTECODE_OP(op) \
 | |
|     op,
 | |
|         ENUMERATE_BYTECODE_OPS(__BYTECODE_OP)
 | |
| #undef __BYTECODE_OP
 | |
|     };
 | |
| 
 | |
|     bool is_terminator() const;
 | |
|     Type type() const { return m_type; }
 | |
|     size_t length() const;
 | |
|     String to_string(Bytecode::Executable const&) const;
 | |
|     ThrowCompletionOr<void> execute(Bytecode::Interpreter&) const;
 | |
|     void replace_references(BasicBlock const&, BasicBlock const&);
 | |
|     static void destroy(Instruction&);
 | |
| 
 | |
| protected:
 | |
|     explicit Instruction(Type type)
 | |
|         : m_type(type)
 | |
|     {
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     Type m_type {};
 | |
| };
 | |
| 
 | |
| class InstructionStreamIterator {
 | |
| public:
 | |
|     explicit InstructionStreamIterator(ReadonlyBytes bytes)
 | |
|         : m_bytes(bytes)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     size_t offset() const { return m_offset; }
 | |
|     bool at_end() const { return m_offset >= m_bytes.size(); }
 | |
|     void jump(size_t offset)
 | |
|     {
 | |
|         VERIFY(offset <= m_bytes.size());
 | |
|         m_offset = offset;
 | |
|     }
 | |
| 
 | |
|     Instruction const& operator*() const { return dereference(); }
 | |
| 
 | |
|     ALWAYS_INLINE void operator++()
 | |
|     {
 | |
|         VERIFY(!at_end());
 | |
|         m_offset += dereference().length();
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     Instruction const& dereference() const { return *reinterpret_cast<Instruction const*>(m_bytes.data() + offset()); }
 | |
| 
 | |
|     ReadonlyBytes m_bytes;
 | |
|     size_t m_offset { 0 };
 | |
| };
 | |
| 
 | |
| }
 |