| /* |
| * Copyright (C) 1999-2001 Harri Porten (porten@kde.org) |
| * Copyright (C) 2003-2021 Apple Inc. All rights reserved. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public License |
| * along with this library; see the file COPYING.LIB. If not, write to |
| * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| * |
| */ |
| |
| #pragma once |
| |
| #include "CallFrame.h" |
| #include <wtf/CheckedArithmetic.h> |
| #include <wtf/ForbidHeapAllocation.h> |
| #include <wtf/HashSet.h> |
| |
| namespace JSC { |
| |
| class alignas(alignof(EncodedJSValue)) MarkedArgumentBufferBase : public RecordOverflow { |
| WTF_MAKE_NONCOPYABLE(MarkedArgumentBufferBase); |
| WTF_MAKE_NONMOVABLE(MarkedArgumentBufferBase); |
| WTF_FORBID_HEAP_ALLOCATION; |
| friend class VM; |
| friend class ArgList; |
| |
| public: |
| using Base = RecordOverflow; |
| typedef HashSet<MarkedArgumentBufferBase*> ListSet; |
| |
| ~MarkedArgumentBufferBase() |
| { |
| ASSERT(!m_needsOverflowCheck); |
| if (m_markSet) |
| m_markSet->remove(this); |
| |
| if (EncodedJSValue* base = mallocBase()) |
| Gigacage::free(Gigacage::JSValue, base); |
| } |
| |
| size_t size() const { return m_size; } |
| bool isEmpty() const { return !m_size; } |
| |
| JSValue at(int i) const |
| { |
| if (i >= m_size) |
| return jsUndefined(); |
| |
| return JSValue::decode(slotFor(i)); |
| } |
| |
| void clear() |
| { |
| ASSERT(!m_needsOverflowCheck); |
| clearOverflow(); |
| m_size = 0; |
| } |
| |
| enum OverflowCheckAction { |
| CrashOnOverflow, |
| WillCheckLater |
| }; |
| template<OverflowCheckAction action> |
| void appendWithAction(JSValue v) |
| { |
| ASSERT(m_size <= m_capacity); |
| if (m_size == m_capacity || mallocBase()) { |
| slowAppend(v); |
| if (action == CrashOnOverflow) |
| RELEASE_ASSERT(!hasOverflowed()); |
| return; |
| } |
| |
| slotFor(m_size) = JSValue::encode(v); |
| ++m_size; |
| } |
| void append(JSValue v) { appendWithAction<WillCheckLater>(v); } |
| void appendWithCrashOnOverflow(JSValue v) { appendWithAction<CrashOnOverflow>(v); } |
| |
| void removeLast() |
| { |
| ASSERT(m_size); |
| m_size--; |
| } |
| |
| JSValue last() |
| { |
| ASSERT(m_size); |
| return JSValue::decode(slotFor(m_size - 1)); |
| } |
| |
| JSValue takeLast() |
| { |
| JSValue result = last(); |
| removeLast(); |
| return result; |
| } |
| |
| template<typename Visitor> static void markLists(Visitor&, ListSet&); |
| |
| void ensureCapacity(size_t requestedCapacity) |
| { |
| if (requestedCapacity > static_cast<size_t>(m_capacity)) |
| slowEnsureCapacity(requestedCapacity); |
| } |
| |
| bool hasOverflowed() |
| { |
| clearNeedsOverflowCheck(); |
| return Base::hasOverflowed(); |
| } |
| |
| void overflowCheckNotNeeded() { clearNeedsOverflowCheck(); } |
| |
| template<typename Functor> |
| void fill(size_t count, const Functor& func) |
| { |
| ASSERT(!m_size); |
| ensureCapacity(count); |
| if (Base::hasOverflowed()) |
| return; |
| m_size = count; |
| func(reinterpret_cast<JSValue*>(&slotFor(0))); |
| } |
| |
| protected: |
| // Constructor for a read-write list, to which you may append values. |
| // FIXME: Remove all clients of this API, then remove this API. |
| MarkedArgumentBufferBase(size_t capacity) |
| : m_size(0) |
| , m_capacity(capacity) |
| , m_buffer(inlineBuffer()) |
| , m_markSet(nullptr) |
| { |
| } |
| |
| EncodedJSValue* inlineBuffer() |
| { |
| return bitwise_cast<EncodedJSValue*>(bitwise_cast<uint8_t*>(this) + sizeof(MarkedArgumentBufferBase)); |
| } |
| |
| private: |
| void expandCapacity(); |
| void expandCapacity(int newCapacity); |
| void slowEnsureCapacity(size_t requestedCapacity); |
| |
| void addMarkSet(JSValue); |
| |
| JS_EXPORT_PRIVATE void slowAppend(JSValue); |
| |
| EncodedJSValue& slotFor(int item) const |
| { |
| return m_buffer[item]; |
| } |
| |
| EncodedJSValue* mallocBase() |
| { |
| if (m_buffer == inlineBuffer()) |
| return nullptr; |
| return &slotFor(0); |
| } |
| |
| #if ASSERT_ENABLED |
| void setNeedsOverflowCheck() { m_needsOverflowCheck = true; } |
| void clearNeedsOverflowCheck() { m_needsOverflowCheck = false; } |
| |
| bool m_needsOverflowCheck { false }; |
| #else |
| void setNeedsOverflowCheck() { } |
| void clearNeedsOverflowCheck() { } |
| #endif // ASSERT_ENABLED |
| int m_size; |
| int m_capacity; |
| EncodedJSValue* m_buffer; |
| ListSet* m_markSet; |
| }; |
| |
| template<size_t passedInlineCapacity = 8> |
| class MarkedArgumentBufferWithSize : public MarkedArgumentBufferBase { |
| public: |
| static constexpr size_t inlineCapacity = passedInlineCapacity; |
| |
| MarkedArgumentBufferWithSize() |
| : MarkedArgumentBufferBase(inlineCapacity) |
| { |
| ASSERT(inlineBuffer() == m_inlineBuffer); |
| } |
| |
| private: |
| EncodedJSValue m_inlineBuffer[inlineCapacity] { }; |
| }; |
| |
| using MarkedArgumentBuffer = MarkedArgumentBufferWithSize<>; |
| |
| class ArgList { |
| WTF_MAKE_FAST_ALLOCATED; |
| friend class Interpreter; |
| friend class JIT; |
| public: |
| ArgList() |
| : m_args(nullptr) |
| , m_argCount(0) |
| { |
| } |
| |
| ArgList(CallFrame* callFrame) |
| : m_args(reinterpret_cast<JSValue*>(&callFrame[CallFrame::argumentOffset(0)])) |
| , m_argCount(callFrame->argumentCount()) |
| { |
| } |
| |
| ArgList(const MarkedArgumentBuffer& args) |
| : m_args(reinterpret_cast<JSValue*>(args.m_buffer)) |
| , m_argCount(args.size()) |
| { |
| } |
| |
| JSValue at(int i) const |
| { |
| if (i >= m_argCount) |
| return jsUndefined(); |
| return m_args[i]; |
| } |
| |
| bool isEmpty() const { return !m_argCount; } |
| size_t size() const { return m_argCount; } |
| |
| JS_EXPORT_PRIVATE void getSlice(int startIndex, ArgList& result) const; |
| |
| private: |
| JSValue* data() const { return m_args; } |
| |
| JSValue* m_args; |
| int m_argCount; |
| }; |
| |
| } // namespace JSC |