diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 19:33:14 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 19:33:14 +0000 |
commit | 36d22d82aa202bb199967e9512281e9a53db42c9 (patch) | |
tree | 105e8c98ddea1c1e4784a60a5a6410fa416be2de /dom/encoding/TextDecoder.cpp | |
parent | Initial commit. (diff) | |
download | firefox-esr-36d22d82aa202bb199967e9512281e9a53db42c9.tar.xz firefox-esr-36d22d82aa202bb199967e9512281e9a53db42c9.zip |
Adding upstream version 115.7.0esr.upstream/115.7.0esr
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'dom/encoding/TextDecoder.cpp')
-rw-r--r-- | dom/encoding/TextDecoder.cpp | 128 |
1 files changed, 128 insertions, 0 deletions
diff --git a/dom/encoding/TextDecoder.cpp b/dom/encoding/TextDecoder.cpp new file mode 100644 index 0000000000..de49bd0122 --- /dev/null +++ b/dom/encoding/TextDecoder.cpp @@ -0,0 +1,128 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=8 sts=2 et sw=2 tw=80: */ +/* 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/TextDecoder.h" +#include "mozilla/dom/UnionTypes.h" +#include "mozilla/Encoding.h" +#include "mozilla/UniquePtrExtensions.h" +#include "nsContentUtils.h" + +#include <stdint.h> + +namespace mozilla::dom { + +void TextDecoder::Init(const nsAString& aLabel, + const TextDecoderOptions& aOptions, ErrorResult& aRv) { + // Let encoding be the result of getting an encoding from label. + // If encoding is failure or replacement, throw a RangeError + // (https://encoding.spec.whatwg.org/#dom-textdecoder). + const Encoding* encoding = Encoding::ForLabelNoReplacement(aLabel); + if (!encoding) { + NS_ConvertUTF16toUTF8 label(aLabel); + label.Trim(" \t\n\f\r"); + aRv.ThrowRangeError<MSG_ENCODING_NOT_SUPPORTED>(label); + return; + } + InitWithEncoding(WrapNotNull(encoding), aOptions); +} + +void TextDecoder::InitWithEncoding(NotNull<const Encoding*> aEncoding, + const TextDecoderOptions& aOptions) { + aEncoding->Name(mEncoding); + // Store the flags passed via our options dictionary. + mFatal = aOptions.mFatal; + mIgnoreBOM = aOptions.mIgnoreBOM; + + // Create a decoder object for mEncoding. + if (mIgnoreBOM) { + mDecoder = aEncoding->NewDecoderWithoutBOMHandling(); + } else { + mDecoder = aEncoding->NewDecoderWithBOMRemoval(); + } +} + +void TextDecoderCommon::DecodeNative(Span<const uint8_t> aInput, + const bool aStream, + nsAString& aOutDecodedString, + ErrorResult& aRv) { + aOutDecodedString.Truncate(); + + CheckedInt<nsAString::size_type> needed = + mDecoder->MaxUTF16BufferLength(aInput.Length()); + if (!needed.isValid()) { + aRv.Throw(NS_ERROR_OUT_OF_MEMORY); + return; + } + + auto output = aOutDecodedString.GetMutableData(needed.value(), fallible); + if (!output) { + aRv.Throw(NS_ERROR_OUT_OF_MEMORY); + return; + } + + uint32_t result; + size_t read; + size_t written; + if (mFatal) { + std::tie(result, read, written) = + mDecoder->DecodeToUTF16WithoutReplacement(aInput, *output, !aStream); + if (result != kInputEmpty) { + aRv.ThrowTypeError<MSG_DOM_DECODING_FAILED>(); + return; + } + } else { + std::tie(result, read, written, std::ignore) = + mDecoder->DecodeToUTF16(aInput, *output, !aStream); + } + MOZ_ASSERT(result == kInputEmpty); + MOZ_ASSERT(read == aInput.Length()); + MOZ_ASSERT(written <= aOutDecodedString.Length()); + + if (!aOutDecodedString.SetLength(written, fallible)) { + aRv.Throw(NS_ERROR_OUT_OF_MEMORY); + return; + } + + // If the internal streaming flag of the decoder object is not set, + // then reset the encoding algorithm state to the default values + if (!aStream) { + if (mIgnoreBOM) { + mDecoder->Encoding()->NewDecoderWithoutBOMHandlingInto(*mDecoder); + } else { + mDecoder->Encoding()->NewDecoderWithBOMRemovalInto(*mDecoder); + } + } +} + +void TextDecoder::Decode(const Optional<ArrayBufferViewOrArrayBuffer>& aBuffer, + const TextDecodeOptions& aOptions, + nsAString& aOutDecodedString, ErrorResult& aRv) { + if (!aBuffer.WasPassed()) { + DecodeNative(nullptr, aOptions.mStream, aOutDecodedString, aRv); + return; + } + const ArrayBufferViewOrArrayBuffer& buf = aBuffer.Value(); + uint8_t* data; + uint32_t length; + if (buf.IsArrayBufferView()) { + buf.GetAsArrayBufferView().ComputeState(); + data = buf.GetAsArrayBufferView().Data(); + length = buf.GetAsArrayBufferView().Length(); + } else { + MOZ_ASSERT(buf.IsArrayBuffer()); + buf.GetAsArrayBuffer().ComputeState(); + data = buf.GetAsArrayBuffer().Data(); + length = buf.GetAsArrayBuffer().Length(); + } + DecodeNative(Span(data, length), aOptions.mStream, aOutDecodedString, aRv); +} + +void TextDecoderCommon::GetEncoding(nsAString& aEncoding) { + CopyASCIItoUTF16(mEncoding, aEncoding); + nsContentUtils::ASCIIToLower(aEncoding); +} + +} // namespace mozilla::dom |