diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 17:32:43 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 17:32:43 +0000 |
commit | 6bf0a5cb5034a7e684dcc3500e841785237ce2dd (patch) | |
tree | a68f146d7fa01f0134297619fbe7e33db084e0aa /image/SurfacePipe.cpp | |
parent | Initial commit. (diff) | |
download | thunderbird-upstream.tar.xz thunderbird-upstream.zip |
Adding upstream version 1:115.7.0.upstream/1%115.7.0upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
-rw-r--r-- | image/SurfacePipe.cpp | 162 |
1 files changed, 162 insertions, 0 deletions
diff --git a/image/SurfacePipe.cpp b/image/SurfacePipe.cpp new file mode 100644 index 0000000000..2e4a86daef --- /dev/null +++ b/image/SurfacePipe.cpp @@ -0,0 +1,162 @@ +/* -*- 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 "SurfacePipe.h" + +#include <algorithm> // for min + +#include "Decoder.h" + +namespace mozilla { +namespace image { + +using namespace gfx; + +using std::min; + +Maybe<SurfaceInvalidRect> AbstractSurfaceSink::TakeInvalidRect() { + if (mInvalidRect.IsEmpty()) { + return Nothing(); + } + + SurfaceInvalidRect invalidRect; + invalidRect.mInputSpaceRect = invalidRect.mOutputSpaceRect = mInvalidRect; + + // Forget about the invalid rect we're returning. + mInvalidRect = OrientedIntRect(); + + return Some(invalidRect); +} + +uint8_t* AbstractSurfaceSink::DoResetToFirstRow() { + mRow = 0; + return GetRowPointer(); +} + +uint8_t* SurfaceSink::DoAdvanceRowFromBuffer(const uint8_t* aInputRow) { + CopyInputRow(aInputRow); + return DoAdvanceRow(); +} + +uint8_t* SurfaceSink::DoAdvanceRow() { + if (mRow >= uint32_t(InputSize().height)) { + return nullptr; + } + + // If we're vertically flipping the output, we need to flip the invalid rect. + // Since we're dealing with an axis-aligned rect, only the y coordinate needs + // to change. + int32_t invalidY = mFlipVertically ? InputSize().height - (mRow + 1) : mRow; + mInvalidRect.UnionRect(mInvalidRect, + OrientedIntRect(0, invalidY, InputSize().width, 1)); + + mRow = min(uint32_t(InputSize().height), mRow + 1); + + return mRow < uint32_t(InputSize().height) ? GetRowPointer() : nullptr; +} + +nsresult SurfaceSink::Configure(const SurfaceConfig& aConfig) { + IntSize surfaceSize = aConfig.mOutputSize; + + // Allocate the frame. + // XXX(seth): Once every Decoder subclass uses SurfacePipe, we probably want + // to allocate the frame directly here and get rid of Decoder::AllocateFrame + // altogether. + nsresult rv = aConfig.mDecoder->AllocateFrame(surfaceSize, aConfig.mFormat, + aConfig.mAnimParams); + if (NS_FAILED(rv)) { + return rv; + } + + mImageData = aConfig.mDecoder->mImageData; + mImageDataLength = aConfig.mDecoder->mImageDataLength; + mFlipVertically = aConfig.mFlipVertically; + + MOZ_ASSERT(mImageData); + MOZ_ASSERT(uint64_t(mImageDataLength) == uint64_t(surfaceSize.width) * + uint64_t(surfaceSize.height) * + sizeof(uint32_t)); + + ConfigureFilter(surfaceSize, sizeof(uint32_t)); + return NS_OK; +} + +uint8_t* SurfaceSink::GetRowPointer() const { + // If we're flipping vertically, reverse the order in which we traverse the + // rows. + uint32_t row = mFlipVertically ? InputSize().height - (mRow + 1) : mRow; + + uint8_t* rowPtr = mImageData + row * InputSize().width * sizeof(uint32_t); + + MOZ_ASSERT(rowPtr >= mImageData); + MOZ_ASSERT(rowPtr < mImageData + mImageDataLength); + MOZ_ASSERT(rowPtr + InputSize().width * sizeof(uint32_t) <= + mImageData + mImageDataLength); + + return rowPtr; +} + +uint8_t* ReorientSurfaceSink::DoAdvanceRowFromBuffer(const uint8_t* aInputRow) { + if (mRow >= uint32_t(InputSize().height)) { + return nullptr; + } + + IntRect dirty = mReorientFn(aInputRow, mRow, mImageData, mSurfaceSize, + mSurfaceSize.width * sizeof(uint32_t)); + auto orientedDirty = OrientedIntRect::FromUnknownRect(dirty); + mInvalidRect.UnionRect(mInvalidRect, orientedDirty); + + mRow = min(uint32_t(InputSize().height), mRow + 1); + + return mRow < uint32_t(InputSize().height) ? GetRowPointer() : nullptr; +} + +uint8_t* ReorientSurfaceSink::DoAdvanceRow() { + return DoAdvanceRowFromBuffer(mBuffer.get()); +} + +nsresult ReorientSurfaceSink::Configure(const ReorientSurfaceConfig& aConfig) { + mSurfaceSize = aConfig.mOutputSize.ToUnknownSize(); + + // Allocate the frame. + // XXX(seth): Once every Decoder subclass uses SurfacePipe, we probably want + // to allocate the frame directly here and get rid of Decoder::AllocateFrame + // altogether. + nsresult rv = + aConfig.mDecoder->AllocateFrame(mSurfaceSize, aConfig.mFormat, Nothing()); + if (NS_FAILED(rv)) { + return rv; + } + + // The filters above us need the unoriented size as the input. + auto inputSize = + aConfig.mOrientation.ToUnoriented(aConfig.mOutputSize).ToUnknownSize(); + mBuffer.reset(new (fallible) uint8_t[inputSize.width * sizeof(uint32_t)]); + if (MOZ_UNLIKELY(!mBuffer)) { + return NS_ERROR_OUT_OF_MEMORY; + } + + memset(mBuffer.get(), 0xFF, inputSize.width * sizeof(uint32_t)); + + mReorientFn = ReorientRow(aConfig.mOrientation); + MOZ_ASSERT(mReorientFn); + + mImageData = aConfig.mDecoder->mImageData; + mImageDataLength = aConfig.mDecoder->mImageDataLength; + + MOZ_ASSERT(mImageData); + MOZ_ASSERT(uint64_t(mImageDataLength) == uint64_t(mSurfaceSize.width) * + uint64_t(mSurfaceSize.height) * + sizeof(uint32_t)); + + ConfigureFilter(inputSize, sizeof(uint32_t)); + return NS_OK; +} + +uint8_t* ReorientSurfaceSink::GetRowPointer() const { return mBuffer.get(); } + +} // namespace image +} // namespace mozilla |