|  | /* | 
|  | * Copyright (C) 2016-2017 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. ``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 | 
|  | * 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. | 
|  | */ | 
|  |  | 
|  | #pragma once | 
|  |  | 
|  | #if ENABLE(WEBASSEMBLY) | 
|  |  | 
|  | #include "ArrayBuffer.h" | 
|  |  | 
|  | #include <limits.h> | 
|  |  | 
|  | namespace WTF { | 
|  | class PrintStream; | 
|  | } | 
|  |  | 
|  | namespace JSC { namespace Wasm { | 
|  |  | 
|  | class PageCount { | 
|  |  | 
|  | public: | 
|  | PageCount() | 
|  | : m_pageCount(UINT_MAX) | 
|  | { | 
|  | static_assert(maxPageCount < UINT_MAX, "We rely on this here."); | 
|  | } | 
|  |  | 
|  | PageCount(uint32_t pageCount) | 
|  | : m_pageCount(pageCount) | 
|  | { } | 
|  |  | 
|  | void dump(WTF::PrintStream&) const; | 
|  |  | 
|  | uint64_t bytes() const { return static_cast<uint64_t>(m_pageCount) * static_cast<uint64_t>(pageSize); } | 
|  | uint32_t pageCount() const { return m_pageCount; } | 
|  |  | 
|  | static bool isValid(uint32_t pageCount) | 
|  | { | 
|  | return pageCount <= maxPageCount; | 
|  | } | 
|  |  | 
|  | bool isValid() const | 
|  | { | 
|  | return isValid(m_pageCount); | 
|  | } | 
|  |  | 
|  | static PageCount fromBytes(uint64_t bytes) | 
|  | { | 
|  | RELEASE_ASSERT(bytes % pageSize == 0); | 
|  | uint32_t numPages = bytes / pageSize; | 
|  | RELEASE_ASSERT(PageCount::isValid(numPages)); | 
|  | return PageCount(numPages); | 
|  | } | 
|  |  | 
|  | static PageCount max() | 
|  | { | 
|  | return PageCount(maxPageCount); | 
|  | } | 
|  |  | 
|  | explicit operator bool() const | 
|  | { | 
|  | return m_pageCount != UINT_MAX; | 
|  | } | 
|  |  | 
|  | bool operator<(const PageCount& other) const { return m_pageCount < other.m_pageCount; } | 
|  | bool operator>(const PageCount& other) const { return m_pageCount > other.m_pageCount; } | 
|  | bool operator>=(const PageCount& other) const { return m_pageCount >= other.m_pageCount; } | 
|  | PageCount operator+(const PageCount& other) const | 
|  | { | 
|  | if (sumOverflows<uint32_t>(m_pageCount, other.m_pageCount)) | 
|  | return PageCount(); | 
|  | uint32_t newCount = m_pageCount + other.m_pageCount; | 
|  | if (!PageCount::isValid(newCount)) | 
|  | return PageCount(); | 
|  | return PageCount(newCount); | 
|  | } | 
|  |  | 
|  | static constexpr uint32_t pageSize = 64 * KB; | 
|  | private: | 
|  | // The spec requires we are able to instantiate a memory with a *maximum* size of 64K pages. | 
|  | // This does not mean the memory can necessarily grow that big, and where the | 
|  | // MAX_ARRAY_BUFFER_SIZE is smaller (e.g.: on 32-bit platforms), trying to grow the memory | 
|  | // that large will fail, which is acceptable according to the spec. Nevertheless, we should | 
|  | // be able to parse such a memory and instantiate it with a smaller initial size. | 
|  | static constexpr uint32_t maxPageCount = std::max<uint32_t>(64*1024, MAX_ARRAY_BUFFER_SIZE / static_cast<uint64_t>(pageSize)); | 
|  |  | 
|  | uint32_t m_pageCount; | 
|  | }; | 
|  |  | 
|  | } } // namespace JSC::Wasm | 
|  |  | 
|  | #endif // ENABLE(WEBASSEMBLY) |