d170c0fb48
Relanding https://skia-review.googlesource.com/c/14105/ in pieces to try to diagnose problems with the Chrome roll. Bug: skia:6535 Change-Id: Iefe4825b9ce9be999baeec8ab7ae6651f1caf451 Reviewed-on: https://skia-review.googlesource.com/14860 Reviewed-by: Mike Reed <reed@google.com> Commit-Queue: Matt Sarett <msarett@google.com>
192 lines
6.7 KiB
C++
192 lines
6.7 KiB
C++
/*
|
|
* Copyright 2008 The Android Open Source Project
|
|
*
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*/
|
|
|
|
#ifndef SkPixelRef_DEFINED
|
|
#define SkPixelRef_DEFINED
|
|
|
|
#include "../private/SkAtomics.h"
|
|
#include "../private/SkMutex.h"
|
|
#include "../private/SkTDArray.h"
|
|
#include "SkBitmap.h"
|
|
#include "SkFilterQuality.h"
|
|
#include "SkImageInfo.h"
|
|
#include "SkPixmap.h"
|
|
#include "SkRefCnt.h"
|
|
#include "SkSize.h"
|
|
#include "SkString.h"
|
|
|
|
class SkColorTable;
|
|
struct SkIRect;
|
|
|
|
class GrTexture;
|
|
class SkDiscardableMemory;
|
|
|
|
/** \class SkPixelRef
|
|
|
|
This class is the smart container for pixel memory, and is used with SkBitmap.
|
|
This class can be shared/accessed between multiple threads.
|
|
*/
|
|
class SK_API SkPixelRef : public SkRefCnt {
|
|
public:
|
|
#ifdef SK_SUPPORT_LEGACY_PIXELREF_API
|
|
SkPixelRef(const SkImageInfo&, void* addr, size_t rowBytes, sk_sp<SkColorTable> = nullptr);
|
|
|
|
const SkImageInfo& info() const {
|
|
return fInfo;
|
|
}
|
|
|
|
#ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK
|
|
// This is undefined if there are clients in-flight trying to use us
|
|
void android_only_reset(const SkImageInfo&, size_t rowBytes, sk_sp<SkColorTable>);
|
|
#endif
|
|
|
|
/**
|
|
* Change the info's AlphaType. Note that this does not automatically
|
|
* invalidate the generation ID. If the pixel values themselves have
|
|
* changed, then you must explicitly call notifyPixelsChanged() as well.
|
|
*/
|
|
void changeAlphaType(SkAlphaType at);
|
|
|
|
/**
|
|
* Returns the size (in bytes) of the internally allocated memory.
|
|
* This should be implemented in all serializable SkPixelRef derived classes.
|
|
* SkBitmap::fPixelRefOffset + SkBitmap::getSafeSize() should never overflow this value,
|
|
* otherwise the rendering code may attempt to read memory out of bounds.
|
|
*
|
|
* @return default impl returns 0.
|
|
*/
|
|
virtual size_t getAllocatedSizeInBytes() const { return 0; }
|
|
|
|
#endif
|
|
|
|
SkPixelRef(int width, int height, void* addr, size_t rowBytes, sk_sp<SkColorTable> = nullptr);
|
|
|
|
~SkPixelRef() override;
|
|
|
|
int width() const { return fInfo.width(); }
|
|
int height() const { return fInfo.height(); }
|
|
void* pixels() const { return fPixels; }
|
|
SkColorTable* colorTable() const { return fCTable.get(); }
|
|
size_t rowBytes() const { return fRowBytes; }
|
|
|
|
/** Returns a non-zero, unique value corresponding to the pixels in this
|
|
pixelref. Each time the pixels are changed (and notifyPixelsChanged is
|
|
called), a different generation ID will be returned.
|
|
*/
|
|
uint32_t getGenerationID() const;
|
|
|
|
#ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK
|
|
/** Returns a non-zero, unique value corresponding to this SkPixelRef.
|
|
Unlike the generation ID, this ID remains the same even when the pixels
|
|
are changed. IDs are not reused (until uint32_t wraps), so it is safe
|
|
to consider this ID unique even after this SkPixelRef is deleted.
|
|
|
|
Can be used as a key which uniquely identifies this SkPixelRef
|
|
regardless of changes to its pixels or deletion of this object.
|
|
*/
|
|
uint32_t getStableID() const { return fStableID; }
|
|
#endif
|
|
|
|
/**
|
|
* Call this if you have changed the contents of the pixels. This will in-
|
|
* turn cause a different generation ID value to be returned from
|
|
* getGenerationID().
|
|
*/
|
|
void notifyPixelsChanged();
|
|
|
|
/** Returns true if this pixelref is marked as immutable, meaning that the
|
|
contents of its pixels will not change for the lifetime of the pixelref.
|
|
*/
|
|
bool isImmutable() const { return fMutability != kMutable; }
|
|
|
|
/** Marks this pixelref is immutable, meaning that the contents of its
|
|
pixels will not change for the lifetime of the pixelref. This state can
|
|
be set on a pixelref, but it cannot be cleared once it is set.
|
|
*/
|
|
void setImmutable();
|
|
|
|
// Register a listener that may be called the next time our generation ID changes.
|
|
//
|
|
// We'll only call the listener if we're confident that we are the only SkPixelRef with this
|
|
// generation ID. If our generation ID changes and we decide not to call the listener, we'll
|
|
// never call it: you must add a new listener for each generation ID change. We also won't call
|
|
// the listener when we're certain no one knows what our generation ID is.
|
|
//
|
|
// This can be used to invalidate caches keyed by SkPixelRef generation ID.
|
|
struct GenIDChangeListener {
|
|
virtual ~GenIDChangeListener() {}
|
|
virtual void onChange() = 0;
|
|
};
|
|
|
|
// Takes ownership of listener.
|
|
void addGenIDChangeListener(GenIDChangeListener* listener);
|
|
|
|
// Call when this pixelref is part of the key to a resourcecache entry. This allows the cache
|
|
// to know automatically those entries can be purged when this pixelref is changed or deleted.
|
|
void notifyAddedToCache() {
|
|
fAddedToCache.store(true);
|
|
}
|
|
|
|
virtual SkDiscardableMemory* diagnostic_only_getDiscardable() const { return NULL; }
|
|
|
|
protected:
|
|
// default impl does nothing.
|
|
virtual void onNotifyPixelsChanged();
|
|
|
|
#ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK
|
|
// This is undefined if there are clients in-flight trying to use us
|
|
void android_only_reset(int width, int height, size_t rowBytes, sk_sp<SkColorTable>);
|
|
#endif
|
|
|
|
private:
|
|
// TODO (msarett): After we remove legacy APIs, we should replace |fInfo| with just a width
|
|
// and height.
|
|
const SkImageInfo fInfo;
|
|
sk_sp<SkColorTable> fCTable;
|
|
void* fPixels;
|
|
size_t fRowBytes;
|
|
|
|
// Bottom bit indicates the Gen ID is unique.
|
|
bool genIDIsUnique() const { return SkToBool(fTaggedGenID.load() & 1); }
|
|
mutable SkAtomic<uint32_t> fTaggedGenID;
|
|
|
|
#ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK
|
|
const uint32_t fStableID;
|
|
#endif
|
|
|
|
SkTDArray<GenIDChangeListener*> fGenIDChangeListeners; // pointers are owned
|
|
|
|
// Set true by caches when they cache content that's derived from the current pixels.
|
|
SkAtomic<bool> fAddedToCache;
|
|
|
|
enum {
|
|
kMutable, // PixelRefs begin mutable.
|
|
kTemporarilyImmutable, // Considered immutable, but can revert to mutable.
|
|
kImmutable, // Once set to this state, it never leaves.
|
|
} fMutability : 8; // easily fits inside a byte
|
|
|
|
void needsNewGenID();
|
|
void callGenIDChangeListeners();
|
|
|
|
void setTemporarilyImmutable();
|
|
void restoreMutability();
|
|
friend class SkSurface_Raster; // For the two methods above.
|
|
|
|
friend class SkImage_Raster;
|
|
friend class SkSpecialImage_Raster;
|
|
|
|
void setImmutableWithID(uint32_t genID);
|
|
friend class SkImage_Gpu;
|
|
friend class SkImage_Lazy;
|
|
friend class SkSpecialImage_Gpu;
|
|
friend void SkBitmapCache_setImmutableWithID(SkPixelRef*, uint32_t);
|
|
|
|
typedef SkRefCnt INHERITED;
|
|
};
|
|
|
|
#endif
|