mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-13 07:24:47 +00:00
221 lines
7.8 KiB
C++
221 lines
7.8 KiB
C++
/* 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/. */
|
|
|
|
#ifndef GFX_TILEDLAYERBUFFER_H
|
|
#define GFX_TILEDLAYERBUFFER_H
|
|
|
|
// Debug defines
|
|
//#define GFX_TILEDLAYER_DEBUG_OVERLAY
|
|
//#define GFX_TILEDLAYER_PREF_WARNINGS
|
|
//#define GFX_TILEDLAYER_RETAINING_LOG
|
|
|
|
#include <stdint.h> // for uint16_t, uint32_t
|
|
#include <sys/types.h> // for int32_t
|
|
#include "LayersLogging.h" // for print_stderr
|
|
#include "mozilla/gfx/gfxVars.h"
|
|
#include "mozilla/gfx/Logging.h" // for gfxCriticalError
|
|
#include "mozilla/layers/LayersTypes.h" // for TextureDumpMode
|
|
#include "nsDebug.h" // for NS_ASSERTION
|
|
#include "nsPoint.h" // for nsIntPoint
|
|
#include "nsRect.h" // for mozilla::gfx::IntRect
|
|
#include "nsRegion.h" // for nsIntRegion
|
|
#include "nsTArray.h" // for nsTArray
|
|
|
|
namespace mozilla {
|
|
|
|
struct TileUnit {};
|
|
template<> struct IsPixel<TileUnit> : mozilla::TrueType {};
|
|
|
|
namespace layers {
|
|
|
|
// You can enable all the TILING_LOG print statements by
|
|
// changing the 0 to a 1 in the following #define.
|
|
#define ENABLE_TILING_LOG 0
|
|
|
|
#if ENABLE_TILING_LOG
|
|
# define TILING_LOG(...) printf_stderr(__VA_ARGS__);
|
|
#else
|
|
# define TILING_LOG(...)
|
|
#endif
|
|
|
|
// Normal integer division truncates towards zero,
|
|
// we instead want to floor to hangle negative numbers.
|
|
static inline int floor_div(int a, int b)
|
|
{
|
|
int rem = a % b;
|
|
int div = a/b;
|
|
if (rem == 0) {
|
|
return div;
|
|
} else {
|
|
// If the signs are different substract 1.
|
|
int sub;
|
|
sub = a ^ b;
|
|
// The results of this shift is either 0 or -1.
|
|
sub >>= 8*sizeof(int)-1;
|
|
return div+sub;
|
|
}
|
|
}
|
|
|
|
// Tiles are aligned to a grid with one of the grid points at (0,0) and other
|
|
// grid points spaced evenly in the x- and y-directions by GetTileSize()
|
|
// multiplied by mResolution. GetScaledTileSize() provides convenience for
|
|
// accessing these values.
|
|
//
|
|
// This tile buffer stores a valid region, which defines the areas that have
|
|
// up-to-date content. The contents of tiles within this region will be reused
|
|
// from paint to paint. It also stores the region that was modified in the last
|
|
// paint operation; this is useful when one tiled layer buffer shadows another
|
|
// (as in an off-main-thread-compositing scenario), so that the shadow tiled
|
|
// layer buffer can correctly reflect the updates of the master layer buffer.
|
|
//
|
|
// The associated Tile may be of any type as long as the derived class can
|
|
// validate and return tiles of that type. Tiles will be frequently copied, so
|
|
// the tile type should be a reference or some other type with an efficient
|
|
// copy constructor.
|
|
//
|
|
// The contents of the tile buffer will be rendered at the resolution specified
|
|
// in mResolution, which can be altered with SetResolution. The resolution
|
|
// should always be a factor of the tile length, to avoid tiles covering
|
|
// non-integer amounts of pixels.
|
|
|
|
// Size and Point in number of tiles rather than in pixels
|
|
typedef gfx::IntSizeTyped<TileUnit> TileIntSize;
|
|
typedef gfx::IntPointTyped<TileUnit> TileIntPoint;
|
|
|
|
/**
|
|
* Stores the origin and size of a tile buffer and handles switching between
|
|
* tile indices and tile positions.
|
|
*
|
|
* Tile positions in TileIntPoint take the first tile offset into account which
|
|
* means that two TilesPlacement of the same layer and resolution give tile
|
|
* positions in the same coordinate space (useful when changing the offset and/or
|
|
* size of a tile buffer).
|
|
*/
|
|
struct TilesPlacement {
|
|
// in tiles
|
|
TileIntPoint mFirst;
|
|
TileIntSize mSize;
|
|
|
|
TilesPlacement(int aFirstX, int aFirstY,
|
|
int aRetainedWidth, int aRetainedHeight)
|
|
: mFirst(aFirstX, aFirstY)
|
|
, mSize(aRetainedWidth, aRetainedHeight)
|
|
{}
|
|
|
|
int TileIndex(TileIntPoint aPosition) const {
|
|
return (aPosition.x - mFirst.x) * mSize.height + aPosition.y - mFirst.y;
|
|
}
|
|
|
|
TileIntPoint TilePosition(size_t aIndex) const {
|
|
return TileIntPoint(
|
|
mFirst.x + aIndex / mSize.height,
|
|
mFirst.y + aIndex % mSize.height
|
|
);
|
|
}
|
|
|
|
bool HasTile(TileIntPoint aPosition) const {
|
|
return aPosition.x >= mFirst.x && aPosition.x < mFirst.x + mSize.width &&
|
|
aPosition.y >= mFirst.y && aPosition.y < mFirst.y + mSize.height;
|
|
}
|
|
};
|
|
|
|
|
|
// Given a position i, this function returns the position inside the current tile.
|
|
inline int GetTileStart(int i, int aTileLength) {
|
|
return (i >= 0) ? (i % aTileLength)
|
|
: ((aTileLength - (-i % aTileLength)) %
|
|
aTileLength);
|
|
}
|
|
|
|
// Rounds the given coordinate down to the nearest tile boundary.
|
|
inline int RoundDownToTileEdge(int aX, int aTileLength) { return aX - GetTileStart(aX, aTileLength); }
|
|
|
|
template<typename Derived, typename Tile>
|
|
class TiledLayerBuffer
|
|
{
|
|
public:
|
|
TiledLayerBuffer()
|
|
: mTiles(0, 0, 0, 0)
|
|
, mResolution(1)
|
|
, mTileSize(mozilla::gfx::gfxVars::TileSize())
|
|
{}
|
|
|
|
~TiledLayerBuffer() {}
|
|
|
|
gfx::IntPoint GetTileOffset(TileIntPoint aPosition) const {
|
|
gfx::IntSize scaledTileSize = GetScaledTileSize();
|
|
return gfx::IntPoint(aPosition.x * scaledTileSize.width,
|
|
aPosition.y * scaledTileSize.height) + mTileOrigin;
|
|
}
|
|
|
|
const TilesPlacement& GetPlacement() const { return mTiles; }
|
|
|
|
const gfx::IntSize& GetTileSize() const { return mTileSize; }
|
|
|
|
gfx::IntSize GetScaledTileSize() const { return gfx::IntSize::Round(gfx::Size(mTileSize) / mResolution); }
|
|
|
|
unsigned int GetTileCount() const { return mRetainedTiles.Length(); }
|
|
|
|
Tile& GetTile(size_t i) { return mRetainedTiles[i]; }
|
|
|
|
const nsIntRegion& GetValidRegion() const { return mValidRegion; }
|
|
const nsIntRegion& GetPaintedRegion() const { return mPaintedRegion; }
|
|
void ClearPaintedRegion() { mPaintedRegion.SetEmpty(); }
|
|
|
|
// Get and set draw scaling. mResolution affects the resolution at which the
|
|
// contents of the buffer are drawn. mResolution has no effect on the
|
|
// coordinate space of the valid region, but does affect the size of an
|
|
// individual tile's rect in relation to the valid region.
|
|
// Setting the resolution will invalidate the buffer.
|
|
float GetResolution() const { return mResolution; }
|
|
bool IsLowPrecision() const { return mResolution < 1; }
|
|
|
|
void Dump(std::stringstream& aStream, const char* aPrefix, bool aDumpHtml,
|
|
TextureDumpMode aCompress);
|
|
|
|
protected:
|
|
|
|
nsIntRegion mValidRegion;
|
|
nsIntRegion mPaintedRegion;
|
|
|
|
/**
|
|
* mRetainedTiles is a rectangular buffer of mTiles.mSize.width x mTiles.mSize.height
|
|
* stored as column major with the same origin as mValidRegion.GetBounds().
|
|
* Any tile that does not intersect mValidRegion is a PlaceholderTile.
|
|
* Only the region intersecting with mValidRegion should be read from a tile,
|
|
* another other region is assumed to be uninitialized. The contents of the
|
|
* tiles is scaled by mResolution.
|
|
*/
|
|
nsTArray<Tile> mRetainedTiles;
|
|
TilesPlacement mTiles;
|
|
float mResolution;
|
|
gfx::IntSize mTileSize;
|
|
gfx::IntPoint mTileOrigin;
|
|
};
|
|
|
|
template<typename Derived, typename Tile> void
|
|
TiledLayerBuffer<Derived, Tile>::Dump(std::stringstream& aStream,
|
|
const char* aPrefix,
|
|
bool aDumpHtml,
|
|
TextureDumpMode aCompress)
|
|
{
|
|
for (size_t i = 0; i < mRetainedTiles.Length(); ++i) {
|
|
const TileIntPoint tilePosition = mTiles.TilePosition(i);
|
|
gfx::IntPoint tileOffset = GetTileOffset(tilePosition);
|
|
|
|
aStream << "\n" << aPrefix << "Tile (x=" <<
|
|
tileOffset.x << ", y=" << tileOffset.y << "): ";
|
|
if (!mRetainedTiles[i].IsPlaceholderTile()) {
|
|
mRetainedTiles[i].DumpTexture(aStream, aCompress);
|
|
} else {
|
|
aStream << "empty tile";
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace layers
|
|
} // namespace mozilla
|
|
|
|
#endif // GFX_TILEDLAYERBUFFER_H
|