blob: 2281f6206f1bed22e66df4b4887706eeb668ef9e [file] [log] [blame]
/*
* Copyright 2024 Google LLC.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef SkPngCodecBase_DEFINED
#define SkPngCodecBase_DEFINED
#include <cstddef>
#include <cstdint>
#include <memory>
#include <optional>
#include "include/codec/SkCodec.h"
#include "include/codec/SkEncodedOrigin.h"
#include "include/core/SkImageInfo.h"
#include "include/core/SkRefCnt.h"
#include "include/private/SkEncodedInfo.h"
#include "include/private/base/SkDebug.h"
#include "include/private/base/SkTemplates.h"
class SkColorPalette;
class SkSampler;
class SkStream;
class SkSwizzler;
enum class SkEncodedImageFormat;
template <typename T> class SkSpan;
// This class implements functionality shared between `SkPngCodec` and
// `SkPngRustCodec` (the latter is from `experimental/rust_png`).
class SkPngCodecBase : public SkCodec {
public:
~SkPngCodecBase() override;
static bool isCompatibleColorProfileAndType(const SkEncodedInfo::ICCProfile* profile,
SkEncodedInfo::Color color);
protected:
SkPngCodecBase(SkEncodedInfo&&, std::unique_ptr<SkStream>, SkEncodedOrigin origin);
// Initialize most fields needed by `applyXformRow`.
//
// Each call to `applyXformRow` will transform `frameWidth` pixels
// (which may be less than `dstInfo.width()` when decoding frames that
// depend on earlier frames).
Result initializeXforms(const SkImageInfo& dstInfo, const Options& options, int frameWidth);
// Initialize other fields needed by `applyXformRow`.
//
// Needs to be called *after* (i.e. outside of) `onStartIncrementalDecode`.
void initializeXformParams();
// Transforms a decoded row into the `dstInfo` format that was earlier
// passed to `initializeXforms`.
//
// The first bytes/pixels of `srcRow` will be transformed into the first
// bytes/pixels of `dstRow`. In other words, the transformation ignores
// `fcTL.x_offset` field - the caller should offset `dstRow` if desired
// (it may not be desirable when working with interlaced rows which are
// first transformed into an intermediate buffer).
void applyXformRow(SkSpan<uint8_t> dstRow, SkSpan<const uint8_t> srcRow);
void applyXformRow(void* dstRow, const uint8_t* srcRow);
size_t getEncodedRowBytes() const { return fEncodedRowBytes; }
const SkSwizzler* swizzler() const { return fSwizzler.get(); }
struct PaletteColorEntry {
uint8_t red;
uint8_t green;
uint8_t blue;
};
virtual std::optional<SkSpan<const PaletteColorEntry>> onTryGetPlteChunk() = 0;
virtual std::optional<SkSpan<const uint8_t>> onTryGetTrnsChunk() = 0;
private:
// SkCodec overrides:
SkEncodedImageFormat onGetEncodedFormat() const final;
SkSampler* getSampler(bool createIfNecessary) final;
void allocateStorage(const SkImageInfo& dstInfo);
Result initializeSwizzler(const SkImageInfo& dstInfo,
const Options& options,
bool skipFormatConversion,
int frameWidth);
bool createColorTable(const SkImageInfo& dstInfo);
enum XformMode {
// Requires only a swizzle pass.
kSwizzleOnly_XformMode,
// Requires only a color xform pass.
kColorOnly_XformMode,
// Requires a swizzle and a color xform.
kSwizzleColor_XformMode,
};
XformMode fXformMode;
std::unique_ptr<SkSwizzler> fSwizzler;
skia_private::AutoTMalloc<uint8_t> fStorage;
int fXformWidth = -1;
sk_sp<SkColorPalette> fColorTable;
size_t fEncodedRowBytes = 0; // Size of encoded/source row in bytes.
size_t fDstRowBytes = 0; // Size of destination row in bytes.
std::optional<SkImageInfo> fDstInfoOfPreviousColorTableCreation;
};
#endif // SkPngCodecBase_DEFINED