diff options
Diffstat (limited to '')
-rw-r--r-- | dom/encoding/TextEncoderStream.cpp | 240 |
1 files changed, 240 insertions, 0 deletions
diff --git a/dom/encoding/TextEncoderStream.cpp b/dom/encoding/TextEncoderStream.cpp new file mode 100644 index 0000000000..f64a4bdb7e --- /dev/null +++ b/dom/encoding/TextEncoderStream.cpp @@ -0,0 +1,240 @@ +/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim:set ts=2 sw=2 sts=2 et cindent: */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +#include "mozilla/dom/TextEncoderStream.h" + +#include "js/ArrayBuffer.h" +#include "js/experimental/TypedData.h" +#include "nsIGlobalObject.h" +#include "mozilla/Encoding.h" +#include "mozilla/dom/BindingUtils.h" +#include "mozilla/dom/TextEncoderStreamBinding.h" +#include "mozilla/dom/TransformerCallbackHelpers.h" +#include "mozilla/dom/TransformStream.h" + +namespace mozilla::dom { + +NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE(TextEncoderStream, mGlobal, mStream) +NS_IMPL_CYCLE_COLLECTING_ADDREF(TextEncoderStream) +NS_IMPL_CYCLE_COLLECTING_RELEASE(TextEncoderStream) +NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(TextEncoderStream) + NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY + NS_INTERFACE_MAP_ENTRY(nsISupports) +NS_INTERFACE_MAP_END + +TextEncoderStream::TextEncoderStream(nsISupports* aGlobal, + TransformStream& aStream) + : mGlobal(aGlobal), mStream(&aStream) { +// See the comment in EncodeNative() about why this uses a decoder instead of +// `UTF_8_ENCODING->NewEncoder()`. +// XXX: We have to consciously choose 16LE/BE because we ultimately have to read +// char16_t* as uint8_t*. See the same comment. +#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ + mDecoder = UTF_16LE_ENCODING->NewDecoder(); +#else + mDecoder = UTF_16BE_ENCODING->NewDecoder(); +#endif +} + +TextEncoderStream::~TextEncoderStream() = default; + +JSObject* TextEncoderStream::WrapObject(JSContext* aCx, + JS::Handle<JSObject*> aGivenProto) { + return TextEncoderStream_Binding::Wrap(aCx, this, aGivenProto); +} + +// Note that the most of the encoding algorithm is implemented in +// mozilla::Decoder (see the comment in EncodeNative()), and this is mainly +// about calling it properly. +static void EncodeNative(JSContext* aCx, mozilla::Decoder* aDecoder, + Span<const char16_t> aInput, const bool aFlush, + JS::MutableHandle<JSObject*> aOutputArrayBufferView, + ErrorResult& aRv) { + // XXX: Adjust the length since Decoder always accepts uint8_t (whereas + // Encoder also accepts char16_t, see below). + if (aInput.Length() > SIZE_MAX / 2) { + aRv.Throw(NS_ERROR_OUT_OF_MEMORY); + return; + } + size_t lengthU8 = aInput.Length() * 2; + + CheckedInt<nsAString::size_type> needed = + aDecoder->MaxUTF8BufferLength(lengthU8); + if (!needed.isValid()) { + aRv.Throw(NS_ERROR_OUT_OF_MEMORY); + return; + } + + UniquePtr<uint8_t[], JS::FreePolicy> buffer( + static_cast<uint8_t*>(JS_malloc(aCx, needed.value()))); + if (!buffer) { + aRv.Throw(NS_ERROR_OUT_OF_MEMORY); + return; + } + + mozilla::Span<uint8_t> input((uint8_t*)aInput.data(), lengthU8); + mozilla::Span<uint8_t> output(buffer, needed.value()); + + // This originally wanted to use mozilla::Encoder::Encode() that accepts + // char16_t*, but it lacks the pending-high-surrogate feature to properly + // implement + // https://encoding.spec.whatwg.org/#convert-code-unit-to-scalar-value. + // See also https://github.com/hsivonen/encoding_rs/issues/82 about the + // reasoning. + // XXX: The code is more verbose here since we need to convert to + // uint8_t* which is the only type mozilla::Decoder accepts. + uint32_t result; + size_t read; + size_t written; + std::tie(result, read, written, std::ignore) = + aDecoder->DecodeToUTF8(input, output, aFlush); + MOZ_ASSERT(result == kInputEmpty); + MOZ_ASSERT(read == lengthU8); + MOZ_ASSERT(written <= needed.value()); + + // https://encoding.spec.whatwg.org/#encode-and-enqueue-a-chunk + // Step 4.2.2.1. Let chunk be a Uint8Array object wrapping an ArrayBuffer + // containing output. + JS::Rooted<JSObject*> arrayBuffer( + aCx, JS::NewArrayBufferWithContents(aCx, written, std::move(buffer))); + if (!arrayBuffer.get()) { + JS_ClearPendingException(aCx); + aRv.Throw(NS_ERROR_OUT_OF_MEMORY); + return; + } + aOutputArrayBufferView.set(JS_NewUint8ArrayWithBuffer( + aCx, arrayBuffer, 0, static_cast<int64_t>(written))); + if (!aOutputArrayBufferView.get()) { + aRv.Throw(NS_ERROR_OUT_OF_MEMORY); + return; + } +} + +class TextEncoderStreamAlgorithms : public TransformerAlgorithmsWrapper { + NS_DECL_ISUPPORTS_INHERITED + NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(TextEncoderStreamAlgorithms, + TransformerAlgorithmsBase) + + void SetEncoderStream(TextEncoderStream& aStream) { + mEncoderStream = &aStream; + } + + // The common part of encode-and-enqueue and encode-and-flush. + // https://encoding.spec.whatwg.org/#decode-and-enqueue-a-chunk + MOZ_CAN_RUN_SCRIPT void EncodeAndEnqueue( + JSContext* aCx, const nsAString& aInput, + TransformStreamDefaultController& aController, bool aFlush, + ErrorResult& aRv) { + JS::Rooted<JSObject*> outView(aCx); + // Passing a Decoder for a reason, see the comments in the method. + EncodeNative(aCx, mEncoderStream->Decoder(), aInput, aFlush, &outView, aRv); + + if (JS_GetTypedArrayLength(outView) > 0) { + // Step 4.2.2.2. Enqueue chunk into encoder’s transform. + JS::Rooted<JS::Value> value(aCx, JS::ObjectValue(*outView)); + aController.Enqueue(aCx, value, aRv); + } + } + + // https://encoding.spec.whatwg.org/#dom-textencoderstream + MOZ_CAN_RUN_SCRIPT void TransformCallbackImpl( + JS::Handle<JS::Value> aChunk, + TransformStreamDefaultController& aController, + ErrorResult& aRv) override { + // Step 2. Let transformAlgorithm be an algorithm which takes a chunk + // argument and runs the encode and enqueue a chunk algorithm with this and + // chunk. + + AutoJSAPI jsapi; + if (!jsapi.Init(aController.GetParentObject())) { + aRv.ThrowUnknownError("Internal error"); + return; + } + JSContext* cx = jsapi.cx(); + + // https://encoding.spec.whatwg.org/#encode-and-enqueue-a-chunk + + // Step 1. Let input be the result of converting chunk to a DOMString. + // Step 2. Convert input to an I/O queue of code units. + nsString str; + if (!ConvertJSValueToString(cx, aChunk, eStringify, eStringify, str)) { + aRv.MightThrowJSException(); + aRv.StealExceptionFromJSContext(cx); + return; + } + + EncodeAndEnqueue(cx, str, aController, false, aRv); + } + + // https://encoding.spec.whatwg.org/#dom-textencoderstream + MOZ_CAN_RUN_SCRIPT void FlushCallbackImpl( + TransformStreamDefaultController& aController, + ErrorResult& aRv) override { + // Step 3. Let flushAlgorithm be an algorithm which runs the encode and + // flush algorithm with this. + + AutoJSAPI jsapi; + if (!jsapi.Init(aController.GetParentObject())) { + aRv.ThrowUnknownError("Internal error"); + return; + } + JSContext* cx = jsapi.cx(); + + // The spec manually manages pending high surrogate here, but let's call the + // encoder as it's managed there. + EncodeAndEnqueue(cx, u""_ns, aController, true, aRv); + } + + private: + ~TextEncoderStreamAlgorithms() override = default; + + RefPtr<TextEncoderStream> mEncoderStream; +}; + +NS_IMPL_CYCLE_COLLECTION_INHERITED(TextEncoderStreamAlgorithms, + TransformerAlgorithmsBase, mEncoderStream) +NS_IMPL_ADDREF_INHERITED(TextEncoderStreamAlgorithms, TransformerAlgorithmsBase) +NS_IMPL_RELEASE_INHERITED(TextEncoderStreamAlgorithms, + TransformerAlgorithmsBase) +NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(TextEncoderStreamAlgorithms) +NS_INTERFACE_MAP_END_INHERITING(TransformerAlgorithmsBase) + +// https://encoding.spec.whatwg.org/#dom-textencoderstream +already_AddRefed<TextEncoderStream> TextEncoderStream::Constructor( + const GlobalObject& aGlobal, ErrorResult& aRv) { + // Step 1. Set this’s encoder to an instance of the UTF-8 encoder. + + // Step 2-3 + auto algorithms = MakeRefPtr<TextEncoderStreamAlgorithms>(); + + // Step 4-5 + RefPtr<TransformStream> transformStream = + TransformStream::CreateGeneric(aGlobal, *algorithms, aRv); + if (aRv.Failed()) { + return nullptr; + } + + // Step 6. Set this’s transform to transformStream. + // (Done in the constructor) + auto encoderStream = + MakeRefPtr<TextEncoderStream>(aGlobal.GetAsSupports(), *transformStream); + algorithms->SetEncoderStream(*encoderStream); + return encoderStream.forget(); +} + +ReadableStream* TextEncoderStream::Readable() const { + return mStream->Readable(); +} + +WritableStream* TextEncoderStream::Writable() const { + return mStream->Writable(); +} + +void TextEncoderStream::GetEncoding(nsCString& aRetVal) const { + aRetVal.AssignLiteral("utf-8"); +} + +} // namespace mozilla::dom |