// Copyright (C) 2018 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #include "qv4arraybuffer_p.h" #include "qv4typedarray_p.h" #include "qv4atomics_p.h" #include "qv4symbol_p.h" using namespace QV4; DEFINE_OBJECT_VTABLE(Atomics); void Heap::Atomics::init() { Object::init(); Scope scope(internalClass->engine); ScopedObject m(scope, this); m->defineDefaultProperty(QStringLiteral("add"), QV4::Atomics::method_add, 3); m->defineDefaultProperty(QStringLiteral("and"), QV4::Atomics::method_and, 3); m->defineDefaultProperty(QStringLiteral("compareExchange"), QV4::Atomics::method_compareExchange, 4); m->defineDefaultProperty(QStringLiteral("exchange"), QV4::Atomics::method_exchange, 3); m->defineDefaultProperty(QStringLiteral("isLockFree"), QV4::Atomics::method_isLockFree, 1); m->defineDefaultProperty(QStringLiteral("load"), QV4::Atomics::method_load, 2); m->defineDefaultProperty(QStringLiteral("or"), QV4::Atomics::method_or, 3); m->defineDefaultProperty(QStringLiteral("store"), QV4::Atomics::method_store, 3); m->defineDefaultProperty(QStringLiteral("sub"), QV4::Atomics::method_sub, 3); m->defineDefaultProperty(QStringLiteral("wait"), QV4::Atomics::method_wait, 4); m->defineDefaultProperty(QStringLiteral("wake"), QV4::Atomics::method_wake, 3); m->defineDefaultProperty(QStringLiteral("xor"), QV4::Atomics::method_xor, 3); ScopedString name(scope, scope.engine->newString(QStringLiteral("Atomics"))); m->defineReadonlyConfigurableProperty(scope.engine->symbol_toStringTag(), name); } static SharedArrayBuffer *validateSharedIntegerTypedArray(Scope &scope, const Value &typedArray, bool onlyInt32 = false) { const TypedArray *a = typedArray.as(); if (!a) { scope.engine->throwTypeError(); return nullptr; } TypedArrayType t(a->arrayType()); if (!a->d()->type->atomicLoad || (onlyInt32 && t != TypedArrayType::Int32Array)) { scope.engine->throwTypeError(); return nullptr; } Scoped buffer(scope, a->d()->buffer); if (!buffer->isSharedArrayBuffer()) { scope.engine->throwTypeError(); return nullptr; } Q_ASSERT(!buffer->hasDetachedArrayData()); return buffer; } static int validateAtomicAccess(Scope &scope, const TypedArray &typedArray, const Value &index) { const TypedArray &a = static_cast(typedArray); qint64 idx = index.toIndex(); if (scope.hasException()) return -1; if (idx < 0 || idx >= a.length()) { scope.engine->throwRangeError(QStringLiteral("index out of range.")); return -1; } return static_cast(idx); } ReturnedValue atomicReadModifyWrite(const FunctionObject *f, const Value *argv, int argc, AtomicModifyOps modify) { Scope scope(f); if (!argc) return scope.engine->throwTypeError(); SharedArrayBuffer *buffer = validateSharedIntegerTypedArray(scope, argv[0]); if (!buffer) return Encode::undefined(); const TypedArray &a = static_cast(argv[0]); int index = validateAtomicAccess(scope, a, argc > 1 ? argv[1] : Value::undefinedValue()); if (index < 0) return Encode::undefined(); Value v = Value::fromReturnedValue((argc > 2 ? argv[2] : Value::undefinedValue()).convertedToNumber()); if (scope.hasException()) return Encode::undefined(); int bytesPerElement = a.d()->type->bytesPerElement; int byteOffset = a.d()->byteOffset + index * bytesPerElement; return a.d()->type->atomicModifyOps[modify](buffer->arrayData() + byteOffset, v); } ReturnedValue Atomics::method_add(const FunctionObject *f, const Value *, const Value *argv, int argc) { return atomicReadModifyWrite(f, argv, argc, AtomicAdd); } ReturnedValue Atomics::method_and(const FunctionObject *f, const Value *, const Value *argv, int argc) { return atomicReadModifyWrite(f, argv, argc, AtomicAnd); } ReturnedValue Atomics::method_compareExchange(const FunctionObject *f, const Value *, const Value *argv, int argc) { Scope scope(f); if (!argc) return scope.engine->throwTypeError(); SharedArrayBuffer *buffer = validateSharedIntegerTypedArray(scope, argv[0]); if (!buffer) return Encode::undefined(); const TypedArray &a = static_cast(argv[0]); int index = validateAtomicAccess(scope, a, argc > 1 ? argv[1] : Value::undefinedValue()); if (index < 0) return Encode::undefined(); Value expected = Value::fromReturnedValue((argc > 2 ? argv[2] : Value::undefinedValue()).convertedToNumber()); if (scope.hasException()) return Encode::undefined(); Value v = Value::fromReturnedValue((argc > 3 ? argv[3] : Value::undefinedValue()).convertedToNumber()); if (scope.hasException()) return Encode::undefined(); int bytesPerElement = a.d()->type->bytesPerElement; int byteOffset = a.d()->byteOffset + index * bytesPerElement; return a.d()->type->atomicCompareExchange(buffer->arrayData() + byteOffset, expected, v); } ReturnedValue Atomics::method_exchange(const FunctionObject *f, const Value *, const Value *argv, int argc) { return atomicReadModifyWrite(f, argv, argc, AtomicExchange); } ReturnedValue Atomics::method_isLockFree(const FunctionObject *, const Value *, const Value *argv, int argc) { if (!argc) return Encode(false); double n = argv[0].toInteger(); if (n == 4.) return Encode(true); if (n == 2.) return Encode(QAtomicOps::isTestAndSetNative()); #ifdef Q_ATOMIC_INT8_IS_SUPPORTED if (n == 1.) return Encode(QAtomicOps::isTestAndSetNative()); #endif return Encode(false); } ReturnedValue Atomics::method_load(const FunctionObject *f, const Value *, const Value *argv, int argc) { Scope scope(f); if (!argc) return scope.engine->throwTypeError(); SharedArrayBuffer *buffer = validateSharedIntegerTypedArray(scope, argv[0]); if (!buffer) return Encode::undefined(); const TypedArray &a = static_cast(argv[0]); int index = validateAtomicAccess(scope, a, argc > 1 ? argv[1] : Value::undefinedValue()); if (index < 0) return Encode::undefined(); int bytesPerElement = a.d()->type->bytesPerElement; int byteOffset = a.d()->byteOffset + index * bytesPerElement; return a.d()->type->atomicLoad(buffer->arrayData() + byteOffset); } ReturnedValue Atomics::method_or(const FunctionObject *f, const Value *, const Value *argv, int argc) { return atomicReadModifyWrite(f, argv, argc, AtomicOr); } ReturnedValue Atomics::method_store(const FunctionObject *f, const Value *, const Value *argv, int argc) { Scope scope(f); if (!argc) return scope.engine->throwTypeError(); SharedArrayBuffer *buffer = validateSharedIntegerTypedArray(scope, argv[0]); if (!buffer) return Encode::undefined(); const TypedArray &a = static_cast(argv[0]); int index = validateAtomicAccess(scope, a, argc > 1 ? argv[1] : Value::undefinedValue()); if (index < 0) return Encode::undefined(); Value v = Value::fromReturnedValue((argc > 2 ? argv[2] : Value::undefinedValue()).convertedToNumber()); if (scope.hasException()) return Encode::undefined(); int bytesPerElement = a.d()->type->bytesPerElement; int byteOffset = a.d()->byteOffset + index * bytesPerElement; return a.d()->type->atomicStore(buffer->arrayData() + byteOffset, v); } ReturnedValue Atomics::method_sub(const FunctionObject *f, const Value *, const Value *argv, int argc) { return atomicReadModifyWrite(f, argv, argc, AtomicSub); } ReturnedValue Atomics::method_wait(const FunctionObject *f, const Value *, const Value *, int) { return f->engine()->throwTypeError(); } ReturnedValue Atomics::method_wake(const FunctionObject *f, const Value *, const Value *, int) { return f->engine()->throwTypeError(); } ReturnedValue Atomics::method_xor(const FunctionObject *f, const Value *, const Value *argv, int argc) { return atomicReadModifyWrite(f, argv, argc, AtomicXor); }