| 1 | /* |
| 2 | * Copyright (C) 2013-2019 Apple Inc. All rights reserved. |
| 3 | * |
| 4 | * Redistribution and use in source and binary forms, with or without |
| 5 | * modification, are permitted provided that the following conditions |
| 6 | * are met: |
| 7 | * 1. Redistributions of source code must retain the above copyright |
| 8 | * notice, this list of conditions and the following disclaimer. |
| 9 | * 2. Redistributions in binary form must reproduce the above copyright |
| 10 | * notice, this list of conditions and the following disclaimer in the |
| 11 | * documentation and/or other materials provided with the distribution. |
| 12 | * |
| 13 | * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY |
| 14 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| 15 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| 16 | * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR |
| 17 | * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| 18 | * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| 19 | * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
| 20 | * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
| 21 | * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 22 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| 23 | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 24 | */ |
| 25 | |
| 26 | #include "config.h" |
| 27 | #include "JSArrayBufferView.h" |
| 28 | |
| 29 | #include "GenericTypedArrayViewInlines.h" |
| 30 | #include "JSArrayBuffer.h" |
| 31 | #include "JSCInlines.h" |
| 32 | #include "JSGenericTypedArrayViewInlines.h" |
| 33 | #include "JSTypedArrays.h" |
| 34 | #include "TypeError.h" |
| 35 | #include "TypedArrayController.h" |
| 36 | #include "TypedArrays.h" |
| 37 | #include <wtf/Gigacage.h> |
| 38 | |
| 39 | namespace JSC { |
| 40 | |
| 41 | const ClassInfo JSArrayBufferView::s_info = { |
| 42 | "ArrayBufferView" , &Base::s_info, nullptr, nullptr, CREATE_METHOD_TABLE(JSArrayBufferView) |
| 43 | }; |
| 44 | |
| 45 | String JSArrayBufferView::toStringName(const JSObject*, ExecState*) |
| 46 | { |
| 47 | return "Object"_s ; |
| 48 | } |
| 49 | |
| 50 | JSArrayBufferView::ConstructionContext::ConstructionContext( |
| 51 | Structure* structure, uint32_t length, void* vector) |
| 52 | : m_structure(structure) |
| 53 | , m_vector(vector, length) |
| 54 | , m_length(length) |
| 55 | , m_mode(FastTypedArray) |
| 56 | , m_butterfly(nullptr) |
| 57 | { |
| 58 | ASSERT(vector == removeArrayPtrTag(vector)); |
| 59 | RELEASE_ASSERT(length <= fastSizeLimit); |
| 60 | } |
| 61 | |
| 62 | JSArrayBufferView::ConstructionContext::ConstructionContext( |
| 63 | VM& vm, Structure* structure, uint32_t length, uint32_t elementSize, |
| 64 | InitializationMode mode) |
| 65 | : m_structure(0) |
| 66 | , m_length(length) |
| 67 | , m_butterfly(0) |
| 68 | { |
| 69 | if (length <= fastSizeLimit) { |
| 70 | // Attempt GC allocation. |
| 71 | void* temp; |
| 72 | size_t size = sizeOf(length, elementSize); |
| 73 | temp = vm.primitiveGigacageAuxiliarySpace.allocateNonVirtual(vm, size, nullptr, AllocationFailureMode::ReturnNull); |
| 74 | if (!temp) |
| 75 | return; |
| 76 | |
| 77 | m_structure = structure; |
| 78 | m_vector = VectorType(temp, length); |
| 79 | m_mode = FastTypedArray; |
| 80 | |
| 81 | if (mode == ZeroFill) { |
| 82 | uint64_t* asWords = static_cast<uint64_t*>(vector()); |
| 83 | for (unsigned i = size / sizeof(uint64_t); i--;) |
| 84 | asWords[i] = 0; |
| 85 | } |
| 86 | |
| 87 | return; |
| 88 | } |
| 89 | |
| 90 | // Don't allow a typed array to use more than 2GB. |
| 91 | if (length > static_cast<unsigned>(INT_MAX) / elementSize) |
| 92 | return; |
| 93 | |
| 94 | size_t size = static_cast<size_t>(length) * static_cast<size_t>(elementSize); |
| 95 | m_vector = VectorType(Gigacage::tryMalloc(Gigacage::Primitive, size), length); |
| 96 | if (!m_vector) |
| 97 | return; |
| 98 | if (mode == ZeroFill) |
| 99 | memset(vector(), 0, size); |
| 100 | |
| 101 | vm.heap.reportExtraMemoryAllocated(static_cast<size_t>(length) * elementSize); |
| 102 | |
| 103 | m_structure = structure; |
| 104 | m_mode = OversizeTypedArray; |
| 105 | } |
| 106 | |
| 107 | JSArrayBufferView::ConstructionContext::ConstructionContext( |
| 108 | VM& vm, Structure* structure, RefPtr<ArrayBuffer>&& arrayBuffer, |
| 109 | unsigned byteOffset, unsigned length) |
| 110 | : m_structure(structure) |
| 111 | , m_length(length) |
| 112 | , m_mode(WastefulTypedArray) |
| 113 | { |
| 114 | ASSERT(arrayBuffer->data() == removeArrayPtrTag(arrayBuffer->data())); |
| 115 | m_vector = VectorType(static_cast<uint8_t*>(arrayBuffer->data()) + byteOffset, length); |
| 116 | IndexingHeader ; |
| 117 | indexingHeader.setArrayBuffer(arrayBuffer.get()); |
| 118 | m_butterfly = Butterfly::create(vm, 0, 0, 0, true, indexingHeader, 0); |
| 119 | } |
| 120 | |
| 121 | JSArrayBufferView::ConstructionContext::ConstructionContext( |
| 122 | Structure* structure, RefPtr<ArrayBuffer>&& arrayBuffer, |
| 123 | unsigned byteOffset, unsigned length, DataViewTag) |
| 124 | : m_structure(structure) |
| 125 | , m_length(length) |
| 126 | , m_mode(DataViewMode) |
| 127 | , m_butterfly(0) |
| 128 | { |
| 129 | ASSERT(arrayBuffer->data() == removeArrayPtrTag(arrayBuffer->data())); |
| 130 | m_vector = VectorType(static_cast<uint8_t*>(arrayBuffer->data()) + byteOffset, length); |
| 131 | } |
| 132 | |
| 133 | JSArrayBufferView::JSArrayBufferView(VM& vm, ConstructionContext& context) |
| 134 | : Base(vm, context.structure(), nullptr) |
| 135 | , m_length(context.length()) |
| 136 | , m_mode(context.mode()) |
| 137 | { |
| 138 | setButterfly(vm, context.butterfly()); |
| 139 | ASSERT(context.vector() == removeArrayPtrTag(context.vector())); |
| 140 | m_vector.setWithoutBarrier(context.vector(), m_length); |
| 141 | } |
| 142 | |
| 143 | void JSArrayBufferView::finishCreation(VM& vm) |
| 144 | { |
| 145 | Base::finishCreation(vm); |
| 146 | ASSERT(jsDynamicCast<JSArrayBufferView*>(vm, this)); |
| 147 | switch (m_mode) { |
| 148 | case FastTypedArray: |
| 149 | return; |
| 150 | case OversizeTypedArray: |
| 151 | vm.heap.addFinalizer(this, finalize); |
| 152 | return; |
| 153 | case WastefulTypedArray: |
| 154 | vm.heap.addReference(this, butterfly()->indexingHeader()->arrayBuffer()); |
| 155 | return; |
| 156 | case DataViewMode: |
| 157 | ASSERT(!butterfly()); |
| 158 | vm.heap.addReference(this, jsCast<JSDataView*>(this)->possiblySharedBuffer()); |
| 159 | return; |
| 160 | } |
| 161 | RELEASE_ASSERT_NOT_REACHED(); |
| 162 | } |
| 163 | |
| 164 | void JSArrayBufferView::visitChildren(JSCell* cell, SlotVisitor& visitor) |
| 165 | { |
| 166 | JSArrayBufferView* thisObject = jsCast<JSArrayBufferView*>(cell); |
| 167 | Base::visitChildren(cell, visitor); |
| 168 | |
| 169 | if (thisObject->hasArrayBuffer()) { |
| 170 | WTF::loadLoadFence(); |
| 171 | ArrayBuffer* buffer = thisObject->possiblySharedBuffer(); |
| 172 | RELEASE_ASSERT(buffer); |
| 173 | visitor.addOpaqueRoot(buffer); |
| 174 | } |
| 175 | } |
| 176 | |
| 177 | bool JSArrayBufferView::put( |
| 178 | JSCell* cell, ExecState* exec, PropertyName propertyName, JSValue value, |
| 179 | PutPropertySlot& slot) |
| 180 | { |
| 181 | JSArrayBufferView* thisObject = jsCast<JSArrayBufferView*>(cell); |
| 182 | |
| 183 | if (UNLIKELY(isThisValueAltered(slot, thisObject))) |
| 184 | return ordinarySetSlow(exec, thisObject, propertyName, value, slot.thisValue(), slot.isStrictMode()); |
| 185 | |
| 186 | return Base::put(thisObject, exec, propertyName, value, slot); |
| 187 | } |
| 188 | |
| 189 | ArrayBuffer* JSArrayBufferView::unsharedBuffer() |
| 190 | { |
| 191 | ArrayBuffer* result = possiblySharedBuffer(); |
| 192 | RELEASE_ASSERT(!result->isShared()); |
| 193 | return result; |
| 194 | } |
| 195 | |
| 196 | void JSArrayBufferView::finalize(JSCell* cell) |
| 197 | { |
| 198 | JSArrayBufferView* thisObject = static_cast<JSArrayBufferView*>(cell); |
| 199 | ASSERT(thisObject->m_mode == OversizeTypedArray || thisObject->m_mode == WastefulTypedArray); |
| 200 | if (thisObject->m_mode == OversizeTypedArray) |
| 201 | Gigacage::free(Gigacage::Primitive, thisObject->vector()); |
| 202 | } |
| 203 | |
| 204 | JSArrayBuffer* JSArrayBufferView::unsharedJSBuffer(ExecState* exec) |
| 205 | { |
| 206 | VM& vm = exec->vm(); |
| 207 | return vm.m_typedArrayController->toJS(exec, globalObject(vm), unsharedBuffer()); |
| 208 | } |
| 209 | |
| 210 | JSArrayBuffer* JSArrayBufferView::possiblySharedJSBuffer(ExecState* exec) |
| 211 | { |
| 212 | VM& vm = exec->vm(); |
| 213 | return vm.m_typedArrayController->toJS(exec, globalObject(vm), possiblySharedBuffer()); |
| 214 | } |
| 215 | |
| 216 | void JSArrayBufferView::neuter() |
| 217 | { |
| 218 | auto locker = holdLock(cellLock()); |
| 219 | RELEASE_ASSERT(hasArrayBuffer()); |
| 220 | RELEASE_ASSERT(!isShared()); |
| 221 | m_length = 0; |
| 222 | m_vector.clear(); |
| 223 | } |
| 224 | |
| 225 | static const constexpr size_t ElementSizeData[] = { |
| 226 | #define FACTORY(type) sizeof(typename type ## Adaptor::Type), |
| 227 | FOR_EACH_TYPED_ARRAY_TYPE_EXCLUDING_DATA_VIEW(FACTORY) |
| 228 | #undef FACTORY |
| 229 | }; |
| 230 | |
| 231 | #define FACTORY(type) static_assert(std::is_final<JS ## type ## Array>::value, ""); |
| 232 | FOR_EACH_TYPED_ARRAY_TYPE_EXCLUDING_DATA_VIEW(FACTORY) |
| 233 | #undef FACTORY |
| 234 | |
| 235 | static inline size_t elementSize(JSType type) |
| 236 | { |
| 237 | ASSERT(type >= Int8ArrayType && type <= Float64ArrayType); |
| 238 | return ElementSizeData[type - Int8ArrayType]; |
| 239 | } |
| 240 | |
| 241 | ArrayBuffer* JSArrayBufferView::slowDownAndWasteMemory() |
| 242 | { |
| 243 | ASSERT(m_mode == FastTypedArray || m_mode == OversizeTypedArray); |
| 244 | |
| 245 | // We play this game because we want this to be callable even from places that |
| 246 | // don't have access to ExecState* or the VM, and we only allocate so little |
| 247 | // memory here that it's not necessary to trigger a GC - just accounting what |
| 248 | // we have done is good enough. The sort of bizarre exception to the "allocating |
| 249 | // little memory" is when we transfer a backing buffer into the C heap; this |
| 250 | // will temporarily get counted towards heap footprint (incorrectly, in the case |
| 251 | // of adopting an oversize typed array) but we don't GC here anyway. That's |
| 252 | // almost certainly fine. The worst case is if you created a ton of fast typed |
| 253 | // arrays, and did nothing but caused all of them to slow down and waste memory. |
| 254 | // In that case, your memory footprint will double before the GC realizes what's |
| 255 | // up. But if you do *anything* to trigger a GC watermark check, it will know |
| 256 | // that you *had* done those allocations and it will GC appropriately. |
| 257 | Heap* heap = Heap::heap(this); |
| 258 | VM& vm = *heap->vm(); |
| 259 | DeferGCForAWhile deferGC(*heap); |
| 260 | |
| 261 | RELEASE_ASSERT(!hasIndexingHeader(vm)); |
| 262 | Structure* structure = this->structure(vm); |
| 263 | setButterfly(vm, Butterfly::createOrGrowArrayRight( |
| 264 | butterfly(), vm, this, structure, |
| 265 | structure->outOfLineCapacity(), false, 0, 0)); |
| 266 | |
| 267 | RefPtr<ArrayBuffer> buffer; |
| 268 | unsigned byteLength = m_length * elementSize(type()); |
| 269 | |
| 270 | switch (m_mode) { |
| 271 | case FastTypedArray: |
| 272 | buffer = ArrayBuffer::create(vector(), byteLength); |
| 273 | break; |
| 274 | |
| 275 | case OversizeTypedArray: |
| 276 | // FIXME: consider doing something like "subtracting" from extra memory |
| 277 | // cost, since right now this case will cause the GC to think that we reallocated |
| 278 | // the whole buffer. |
| 279 | buffer = ArrayBuffer::createAdopted(vector(), byteLength); |
| 280 | break; |
| 281 | |
| 282 | default: |
| 283 | RELEASE_ASSERT_NOT_REACHED(); |
| 284 | break; |
| 285 | } |
| 286 | |
| 287 | { |
| 288 | auto locker = holdLock(cellLock()); |
| 289 | butterfly()->indexingHeader()->setArrayBuffer(buffer.get()); |
| 290 | m_vector.setWithoutBarrier(buffer->data(), m_length); |
| 291 | WTF::storeStoreFence(); |
| 292 | m_mode = WastefulTypedArray; |
| 293 | } |
| 294 | heap->addReference(this, buffer.get()); |
| 295 | |
| 296 | return buffer.get(); |
| 297 | } |
| 298 | |
| 299 | // Allocates the full-on native buffer and moves data into the C heap if |
| 300 | // necessary. Note that this never allocates in the GC heap. |
| 301 | RefPtr<ArrayBufferView> JSArrayBufferView::possiblySharedImpl() |
| 302 | { |
| 303 | ArrayBuffer* buffer = possiblySharedBuffer(); |
| 304 | unsigned byteOffset = this->byteOffset(); |
| 305 | unsigned length = this->length(); |
| 306 | switch (type()) { |
| 307 | #define FACTORY(type) \ |
| 308 | case type ## ArrayType: \ |
| 309 | return type ## Array::tryCreate(buffer, byteOffset, length); |
| 310 | FOR_EACH_TYPED_ARRAY_TYPE_EXCLUDING_DATA_VIEW(FACTORY) |
| 311 | #undef FACTORY |
| 312 | case DataViewType: |
| 313 | return DataView::create(buffer, byteOffset, length); |
| 314 | default: |
| 315 | RELEASE_ASSERT_NOT_REACHED(); |
| 316 | return nullptr; |
| 317 | } |
| 318 | } |
| 319 | |
| 320 | } // namespace JSC |
| 321 | |
| 322 | namespace WTF { |
| 323 | |
| 324 | using namespace JSC; |
| 325 | |
| 326 | void printInternal(PrintStream& out, TypedArrayMode mode) |
| 327 | { |
| 328 | switch (mode) { |
| 329 | case FastTypedArray: |
| 330 | out.print("FastTypedArray" ); |
| 331 | return; |
| 332 | case OversizeTypedArray: |
| 333 | out.print("OversizeTypedArray" ); |
| 334 | return; |
| 335 | case WastefulTypedArray: |
| 336 | out.print("WastefulTypedArray" ); |
| 337 | return; |
| 338 | case DataViewMode: |
| 339 | out.print("DataViewMode" ); |
| 340 | return; |
| 341 | } |
| 342 | RELEASE_ASSERT_NOT_REACHED(); |
| 343 | } |
| 344 | |
| 345 | } // namespace WTF |
| 346 | |
| 347 | |