Remove code to push pixmaps to backend textures from GrGpu classes

Replace GrGpu::updateBackendTexture with narrower method that clears
a backend texture.

Creation of data for a solid color compressed texture is lifted up to
GrDirectContext and goes through updateCompressedBackendTexture.

Bug: skia:11786
Change-Id: I1d617623df5e65686f30e57c361a64f78d77f7bd
Reviewed-on: https://skia-review.googlesource.com/c/skia/+/392836
Reviewed-by: Jim Van Verth <jvanverth@google.com>
Reviewed-by: Greg Daniel <egdaniel@google.com>
Commit-Queue: Brian Salomon <bsalomon@google.com>
This commit is contained in:
Brian Salomon 2021-04-08 12:45:58 -04:00 committed by Skia Commit-Bot
parent 04d4969bd2
commit 45889d1269
18 changed files with 586 additions and 567 deletions

View File

@ -280,6 +280,7 @@ public:
SkISize dimensions() const { return {fWidth, fHeight}; } SkISize dimensions() const { return {fWidth, fHeight}; }
int width() const { return fWidth; } int width() const { return fWidth; }
int height() const { return fHeight; } int height() const { return fHeight; }
GrMipmapped mipmapped() const { return fMipmapped; }
bool hasMipmaps() const { return fMipmapped == GrMipmapped::kYes; } bool hasMipmaps() const { return fMipmapped == GrMipmapped::kYes; }
/** deprecated alias of hasMipmaps(). */ /** deprecated alias of hasMipmaps(). */
bool hasMipMaps() const { return this->hasMipmaps(); } bool hasMipMaps() const { return this->hasMipmaps(); }

View File

@ -686,7 +686,7 @@ bool GrConvertPixels(const GrPixmap& dst, const GrCPixmap& src, bool flipY) {
return true; return true;
} }
bool GrClearImage(const GrImageInfo& dstInfo, void* dst, size_t dstRB, SkColor4f color) { bool GrClearImage(const GrImageInfo& dstInfo, void* dst, size_t dstRB, std::array<float, 4> color) {
TRACE_EVENT0("skia.gpu", TRACE_FUNC); TRACE_EVENT0("skia.gpu", TRACE_FUNC);
if (!dstInfo.isValid()) { if (!dstInfo.isValid()) {
@ -700,7 +700,7 @@ bool GrClearImage(const GrImageInfo& dstInfo, void* dst, size_t dstRB, SkColor4f
} }
if (dstInfo.colorType() == GrColorType::kRGB_888) { if (dstInfo.colorType() == GrColorType::kRGB_888) {
// SkRasterPipeline doesn't handle writing to RGB_888. So we handle that specially here. // SkRasterPipeline doesn't handle writing to RGB_888. So we handle that specially here.
uint32_t rgba = color.toBytes_RGBA(); uint32_t rgba = SkColor4f{color[0], color[1], color[2], color[3]}.toBytes_RGBA();
for (int y = 0; y < dstInfo.height(); ++y) { for (int y = 0; y < dstInfo.height(); ++y) {
char* d = static_cast<char*>(dst) + y * dstRB; char* d = static_cast<char*>(dst) + y * dstRB;
for (int x = 0; x < dstInfo.width(); ++x, d += 3) { for (int x = 0; x < dstInfo.width(); ++x, d += 3) {
@ -719,7 +719,7 @@ bool GrClearImage(const GrImageInfo& dstInfo, void* dst, size_t dstRB, SkColor4f
char block[64]; char block[64];
SkArenaAlloc alloc(block, sizeof(block), 1024); SkArenaAlloc alloc(block, sizeof(block), 1024);
SkRasterPipeline_<256> pipeline; SkRasterPipeline_<256> pipeline;
pipeline.append_constant_color(&alloc, color); pipeline.append_constant_color(&alloc, color.data());
switch (lumMode) { switch (lumMode) {
case LumMode::kNone: case LumMode::kNone:
break; break;

View File

@ -39,7 +39,7 @@ void GrFillInCompressedData(SkImage::CompressionType, SkISize dimensions, GrMipm
bool GrConvertPixels(const GrPixmap& dst, const GrCPixmap& src, bool flipY = false); bool GrConvertPixels(const GrPixmap& dst, const GrCPixmap& src, bool flipY = false);
/** Clears the dst image to a constant color. */ /** Clears the dst image to a constant color. */
bool GrClearImage(const GrImageInfo& dstInfo, void* dst, size_t dstRB, SkColor4f color); bool GrClearImage(const GrImageInfo& dstInfo, void* dst, size_t dstRB, std::array<float, 4> color);
#if GR_TEST_UTILS #if GR_TEST_UTILS
/** /**

View File

@ -10,7 +10,9 @@
#include "include/core/SkTraceMemoryDump.h" #include "include/core/SkTraceMemoryDump.h"
#include "include/gpu/GrContextThreadSafeProxy.h" #include "include/gpu/GrContextThreadSafeProxy.h"
#include "src/core/SkAutoMalloc.h"
#include "src/core/SkTaskGroup.h" #include "src/core/SkTaskGroup.h"
#include "src/gpu/GrBackendUtils.h"
#include "src/gpu/GrClientMappedBufferManager.h" #include "src/gpu/GrClientMappedBufferManager.h"
#include "src/gpu/GrContextThreadSafeProxyPriv.h" #include "src/gpu/GrContextThreadSafeProxyPriv.h"
#include "src/gpu/GrDirectContextPriv.h" #include "src/gpu/GrDirectContextPriv.h"
@ -472,26 +474,24 @@ GrBackendTexture GrDirectContext::createBackendTexture(int width, int height,
return this->createBackendTexture(width, height, format, mipMapped, renderable, isProtected); return this->createBackendTexture(width, height, format, mipMapped, renderable, isProtected);
} }
static GrBackendTexture create_and_update_backend_texture( static GrBackendTexture create_and_clear_backend_texture(GrDirectContext* dContext,
GrDirectContext* dContext, SkISize dimensions,
SkISize dimensions, const GrBackendFormat& backendFormat,
const GrBackendFormat& backendFormat, GrMipmapped mipMapped,
GrMipmapped mipMapped, GrRenderable renderable,
GrRenderable renderable, GrProtected isProtected,
GrProtected isProtected, sk_sp<GrRefCntedCallback> finishedCallback,
sk_sp<GrRefCntedCallback> finishedCallback, std::array<float, 4> color) {
const GrGpu::BackendTextureData* data) {
GrGpu* gpu = dContext->priv().getGpu(); GrGpu* gpu = dContext->priv().getGpu();
SkASSERT(data->type() == GrGpu::BackendTextureData::Type::kColor);
GrBackendTexture beTex = gpu->createBackendTexture(dimensions, backendFormat, renderable, GrBackendTexture beTex = gpu->createBackendTexture(dimensions, backendFormat, renderable,
mipMapped, isProtected); mipMapped, isProtected);
if (!beTex.isValid()) { if (!beTex.isValid()) {
return {}; return {};
} }
if (!dContext->priv().getGpu()->updateBackendTexture(beTex, if (!dContext->priv().getGpu()->clearBackendTexture(beTex,
std::move(finishedCallback), std::move(finishedCallback),
data)) { color)) {
dContext->deleteBackendTexture(beTex); dContext->deleteBackendTexture(beTex);
return {}; return {};
} }
@ -555,10 +555,14 @@ GrBackendTexture GrDirectContext::createBackendTexture(int width, int height,
return {}; return {};
} }
GrGpu::BackendTextureData data(color); return create_and_clear_backend_texture(this,
return create_and_update_backend_texture(this, {width, height}, {width, height},
backendFormat, mipMapped, renderable, isProtected, backendFormat,
std::move(finishedCallback), &data); mipMapped,
renderable,
isProtected,
std::move(finishedCallback),
color.array());
} }
GrBackendTexture GrDirectContext::createBackendTexture(int width, int height, GrBackendTexture GrDirectContext::createBackendTexture(int width, int height,
@ -583,10 +587,14 @@ GrBackendTexture GrDirectContext::createBackendTexture(int width, int height,
GrColorType grColorType = SkColorTypeToGrColorType(skColorType); GrColorType grColorType = SkColorTypeToGrColorType(skColorType);
SkColor4f swizzledColor = this->caps()->getWriteSwizzle(format, grColorType).applyTo(color); SkColor4f swizzledColor = this->caps()->getWriteSwizzle(format, grColorType).applyTo(color);
GrGpu::BackendTextureData data(swizzledColor); return create_and_clear_backend_texture(this,
return create_and_update_backend_texture(this, {width, height}, format, {width, height},
mipMapped, renderable, isProtected, format,
std::move(finishedCallback), &data); mipMapped,
renderable,
isProtected,
std::move(finishedCallback),
swizzledColor.array());
} }
GrBackendTexture GrDirectContext::createBackendTexture(const SkPixmap srcData[], GrBackendTexture GrDirectContext::createBackendTexture(const SkPixmap srcData[],
@ -647,8 +655,7 @@ bool GrDirectContext::updateBackendTexture(const GrBackendTexture& backendTextur
return false; return false;
} }
GrGpu::BackendTextureData data(color); return fGpu->clearBackendTexture(backendTexture, std::move(finishedCallback), color.array());
return fGpu->updateBackendTexture(backendTexture, std::move(finishedCallback), &data);
} }
bool GrDirectContext::updateBackendTexture(const GrBackendTexture& backendTexture, bool GrDirectContext::updateBackendTexture(const GrBackendTexture& backendTexture,
@ -670,9 +677,11 @@ bool GrDirectContext::updateBackendTexture(const GrBackendTexture& backendTextur
} }
GrSwizzle swizzle = this->caps()->getWriteSwizzle(format, grColorType); GrSwizzle swizzle = this->caps()->getWriteSwizzle(format, grColorType);
GrGpu::BackendTextureData data(swizzle.applyTo(color)); SkColor4f swizzledColor = swizzle.applyTo(color);
return fGpu->updateBackendTexture(backendTexture, std::move(finishedCallback), &data); return fGpu->clearBackendTexture(backendTexture,
std::move(finishedCallback),
swizzledColor.array());
} }
bool GrDirectContext::updateBackendTexture(const GrBackendTexture& backendTexture, bool GrDirectContext::updateBackendTexture(const GrBackendTexture& backendTexture,
@ -717,7 +726,8 @@ static GrBackendTexture create_and_update_compressed_backend_texture(
GrMipmapped mipMapped, GrMipmapped mipMapped,
GrProtected isProtected, GrProtected isProtected,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const GrGpu::BackendTextureData* data) { const void* data,
size_t size) {
GrGpu* gpu = dContext->priv().getGpu(); GrGpu* gpu = dContext->priv().getGpu();
GrBackendTexture beTex = gpu->createCompressedBackendTexture(dimensions, backendFormat, GrBackendTexture beTex = gpu->createCompressedBackendTexture(dimensions, backendFormat,
@ -727,20 +737,21 @@ static GrBackendTexture create_and_update_compressed_backend_texture(
} }
if (!dContext->priv().getGpu()->updateCompressedBackendTexture( if (!dContext->priv().getGpu()->updateCompressedBackendTexture(
beTex, std::move(finishedCallback), data)) { beTex, std::move(finishedCallback), data, size)) {
dContext->deleteBackendTexture(beTex); dContext->deleteBackendTexture(beTex);
return {}; return {};
} }
return beTex; return beTex;
} }
GrBackendTexture GrDirectContext::createCompressedBackendTexture(int width, int height, GrBackendTexture GrDirectContext::createCompressedBackendTexture(
const GrBackendFormat& backendFormat, int width, int height,
const SkColor4f& color, const GrBackendFormat& backendFormat,
GrMipmapped mipMapped, const SkColor4f& color,
GrProtected isProtected, GrMipmapped mipmapped,
GrGpuFinishedProc finishedProc, GrProtected isProtected,
GrGpuFinishedContext finishedContext) { GrGpuFinishedProc finishedProc,
GrGpuFinishedContext finishedContext) {
TRACE_EVENT0("skia.gpu", TRACE_FUNC); TRACE_EVENT0("skia.gpu", TRACE_FUNC);
auto finishedCallback = GrRefCntedCallback::Make(finishedProc, finishedContext); auto finishedCallback = GrRefCntedCallback::Make(finishedProc, finishedContext);
@ -748,19 +759,35 @@ GrBackendTexture GrDirectContext::createCompressedBackendTexture(int width, int
return {}; return {};
} }
GrGpu::BackendTextureData data(color); SkImage::CompressionType compression = GrBackendFormatToCompressionType(backendFormat);
return create_and_update_compressed_backend_texture(this, {width, height}, if (compression == SkImage::CompressionType::kNone) {
backendFormat, mipMapped, isProtected, return {};
std::move(finishedCallback), &data); }
size_t size = SkCompressedDataSize(compression,
{width, height},
nullptr,
mipmapped == GrMipmapped::kYes);
auto storage = std::make_unique<char[]>(size);
GrFillInCompressedData(compression, {width, height}, mipmapped, storage.get(), color);
return create_and_update_compressed_backend_texture(this,
{width, height},
backendFormat,
mipmapped,
isProtected,
std::move(finishedCallback),
storage.get(),
size);
} }
GrBackendTexture GrDirectContext::createCompressedBackendTexture(int width, int height, GrBackendTexture GrDirectContext::createCompressedBackendTexture(
SkImage::CompressionType compression, int width, int height,
const SkColor4f& color, SkImage::CompressionType compression,
GrMipmapped mipMapped, const SkColor4f& color,
GrProtected isProtected, GrMipmapped mipMapped,
GrGpuFinishedProc finishedProc, GrProtected isProtected,
GrGpuFinishedContext finishedContext) { GrGpuFinishedProc finishedProc,
GrGpuFinishedContext finishedContext) {
TRACE_EVENT0("skia.gpu", TRACE_FUNC); TRACE_EVENT0("skia.gpu", TRACE_FUNC);
GrBackendFormat format = this->compressedBackendFormat(compression); GrBackendFormat format = this->compressedBackendFormat(compression);
return this->createCompressedBackendTexture(width, height, format, color, return this->createCompressedBackendTexture(width, height, format, color,
@ -768,14 +795,15 @@ GrBackendTexture GrDirectContext::createCompressedBackendTexture(int width, int
finishedContext); finishedContext);
} }
GrBackendTexture GrDirectContext::createCompressedBackendTexture(int width, int height, GrBackendTexture GrDirectContext::createCompressedBackendTexture(
const GrBackendFormat& backendFormat, int width, int height,
const void* compressedData, const GrBackendFormat& backendFormat,
size_t dataSize, const void* compressedData,
GrMipmapped mipMapped, size_t dataSize,
GrProtected isProtected, GrMipmapped mipMapped,
GrGpuFinishedProc finishedProc, GrProtected isProtected,
GrGpuFinishedContext finishedContext) { GrGpuFinishedProc finishedProc,
GrGpuFinishedContext finishedContext) {
TRACE_EVENT0("skia.gpu", TRACE_FUNC); TRACE_EVENT0("skia.gpu", TRACE_FUNC);
auto finishedCallback = GrRefCntedCallback::Make(finishedProc, finishedContext); auto finishedCallback = GrRefCntedCallback::Make(finishedProc, finishedContext);
@ -783,19 +811,24 @@ GrBackendTexture GrDirectContext::createCompressedBackendTexture(int width, int
return {}; return {};
} }
GrGpu::BackendTextureData data(compressedData, dataSize); return create_and_update_compressed_backend_texture(this,
return create_and_update_compressed_backend_texture(this, {width, height}, {width, height},
backendFormat, mipMapped, isProtected, backendFormat,
std::move(finishedCallback), &data); mipMapped,
isProtected,
std::move(finishedCallback),
compressedData,
dataSize);
} }
GrBackendTexture GrDirectContext::createCompressedBackendTexture(int width, int height, GrBackendTexture GrDirectContext::createCompressedBackendTexture(
SkImage::CompressionType compression, int width, int height,
const void* data, size_t dataSize, SkImage::CompressionType compression,
GrMipmapped mipMapped, const void* data, size_t dataSize,
GrProtected isProtected, GrMipmapped mipMapped,
GrGpuFinishedProc finishedProc, GrProtected isProtected,
GrGpuFinishedContext finishedContext) { GrGpuFinishedProc finishedProc,
GrGpuFinishedContext finishedContext) {
TRACE_EVENT0("skia.gpu", TRACE_FUNC); TRACE_EVENT0("skia.gpu", TRACE_FUNC);
GrBackendFormat format = this->compressedBackendFormat(compression); GrBackendFormat format = this->compressedBackendFormat(compression);
return this->createCompressedBackendTexture(width, height, format, data, dataSize, mipMapped, return this->createCompressedBackendTexture(width, height, format, data, dataSize, mipMapped,
@ -812,8 +845,25 @@ bool GrDirectContext::updateCompressedBackendTexture(const GrBackendTexture& bac
return false; return false;
} }
GrGpu::BackendTextureData data(color); SkImage::CompressionType compression =
return fGpu->updateCompressedBackendTexture(backendTexture, std::move(finishedCallback), &data); GrBackendFormatToCompressionType(backendTexture.getBackendFormat());
if (compression == SkImage::CompressionType::kNone) {
return {};
}
size_t size = SkCompressedDataSize(compression,
backendTexture.dimensions(),
nullptr,
backendTexture.hasMipmaps());
SkAutoMalloc storage(size);
GrFillInCompressedData(compression,
backendTexture.dimensions(),
backendTexture.mipmapped(),
static_cast<char*>(storage.get()),
color);
return fGpu->updateCompressedBackendTexture(backendTexture,
std::move(finishedCallback),
storage.get(),
size);
} }
bool GrDirectContext::updateCompressedBackendTexture(const GrBackendTexture& backendTexture, bool GrDirectContext::updateCompressedBackendTexture(const GrBackendTexture& backendTexture,
@ -831,9 +881,10 @@ bool GrDirectContext::updateCompressedBackendTexture(const GrBackendTexture& bac
return false; return false;
} }
GrGpu::BackendTextureData data(compressedData, dataSize); return fGpu->updateCompressedBackendTexture(backendTexture,
std::move(finishedCallback),
return fGpu->updateCompressedBackendTexture(backendTexture, std::move(finishedCallback), &data); compressedData,
dataSize);
} }
////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////

View File

@ -754,58 +754,16 @@ void GrGpu::Stats::dumpKeyValuePairs(SkTArray<SkString>* keys, SkTArray<double>*
#endif // GR_GPU_STATS #endif // GR_GPU_STATS
#endif // GR_TEST_UTILS #endif // GR_TEST_UTILS
bool GrGpu::MipMapsAreCorrect(SkISize dimensions, bool GrGpu::CompressedDataIsCorrect(SkISize dimensions,
GrMipmapped mipmapped, SkImage::CompressionType compressionType,
const BackendTextureData* data) { GrMipmapped mipMapped,
int numMipLevels = 1; const void* data,
if (mipmapped == GrMipmapped::kYes) { size_t length) {
numMipLevels = SkMipmap::ComputeLevelCount(dimensions.width(), dimensions.height()) + 1; size_t computedSize = SkCompressedDataSize(compressionType,
} dimensions,
nullptr,
if (!data || data->type() == BackendTextureData::Type::kColor) { mipMapped == GrMipmapped::kYes);
return true; return computedSize == length;
}
if (data->type() == BackendTextureData::Type::kCompressed) {
return false; // This should be going through CompressedDataIsCorrect
}
SkASSERT(data->type() == BackendTextureData::Type::kPixmaps);
if (data->pixmap(0).dimensions() != dimensions) {
return false;
}
GrColorType colorType = data->pixmap(0).colorType();
for (int i = 1; i < numMipLevels; ++i) {
dimensions = {std::max(1, dimensions.width()/2), std::max(1, dimensions.height()/2)};
if (dimensions != data->pixmap(i).dimensions()) {
return false;
}
if (colorType != data->pixmap(i).colorType()) {
return false;
}
}
return true;
}
bool GrGpu::CompressedDataIsCorrect(SkISize dimensions, SkImage::CompressionType compressionType,
GrMipmapped mipMapped, const BackendTextureData* data) {
if (!data || data->type() == BackendTextureData::Type::kColor) {
return true;
}
if (data->type() == BackendTextureData::Type::kPixmaps) {
return false;
}
SkASSERT(data->type() == BackendTextureData::Type::kCompressed);
size_t computedSize = SkCompressedDataSize(compressionType, dimensions,
nullptr, mipMapped == GrMipmapped::kYes);
return computedSize == data->compressedSize();
} }
GrBackendTexture GrGpu::createBackendTexture(SkISize dimensions, GrBackendTexture GrGpu::createBackendTexture(SkISize dimensions,
@ -836,33 +794,18 @@ GrBackendTexture GrGpu::createBackendTexture(SkISize dimensions,
return this->onCreateBackendTexture(dimensions, format, renderable, mipMapped, isProtected); return this->onCreateBackendTexture(dimensions, format, renderable, mipMapped, isProtected);
} }
bool GrGpu::updateBackendTexture(const GrBackendTexture& backendTexture, bool GrGpu::clearBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { std::array<float, 4> color) {
SkASSERT(data);
const GrCaps* caps = this->caps();
if (!backendTexture.isValid()) { if (!backendTexture.isValid()) {
return false; return false;
} }
if (data->type() == BackendTextureData::Type::kPixmaps) {
auto ct = data->pixmap(0).colorType();
if (!caps->areColorTypeAndFormatCompatible(ct, backendTexture.getBackendFormat())) {
return false;
}
}
if (backendTexture.hasMipmaps() && !this->caps()->mipmapSupport()) { if (backendTexture.hasMipmaps() && !this->caps()->mipmapSupport()) {
return false; return false;
} }
GrMipmapped mipmapped = backendTexture.hasMipmaps() ? GrMipmapped::kYes : GrMipmapped::kNo; return this->onClearBackendTexture(backendTexture, std::move(finishedCallback), color);
if (!MipMapsAreCorrect(backendTexture.dimensions(), mipmapped, data)) {
return false;
}
return this->onUpdateBackendTexture(backendTexture, std::move(finishedCallback), data);
} }
GrBackendTexture GrGpu::createCompressedBackendTexture(SkISize dimensions, GrBackendTexture GrGpu::createCompressedBackendTexture(SkISize dimensions,
@ -896,7 +839,8 @@ GrBackendTexture GrGpu::createCompressedBackendTexture(SkISize dimensions,
bool GrGpu::updateCompressedBackendTexture(const GrBackendTexture& backendTexture, bool GrGpu::updateCompressedBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { const void* data,
size_t length) {
SkASSERT(data); SkASSERT(data);
if (!backendTexture.isValid()) { if (!backendTexture.isValid()) {
@ -917,10 +861,16 @@ bool GrGpu::updateCompressedBackendTexture(const GrBackendTexture& backendTextur
GrMipmapped mipMapped = backendTexture.hasMipmaps() ? GrMipmapped::kYes : GrMipmapped::kNo; GrMipmapped mipMapped = backendTexture.hasMipmaps() ? GrMipmapped::kYes : GrMipmapped::kNo;
if (!CompressedDataIsCorrect(backendTexture.dimensions(), compressionType, mipMapped, data)) { if (!CompressedDataIsCorrect(backendTexture.dimensions(),
compressionType,
mipMapped,
data,
length)) {
return false; return false;
} }
return this->onUpdateCompressedBackendTexture(backendTexture, std::move(finishedCallback), return this->onUpdateCompressedBackendTexture(backendTexture,
data); std::move(finishedCallback),
data,
length);
} }

View File

@ -497,58 +497,6 @@ public:
Stats* stats() { return &fStats; } Stats* stats() { return &fStats; }
void dumpJSON(SkJSONWriter*) const; void dumpJSON(SkJSONWriter*) const;
/** Used to initialize a backend texture with either a constant color, pixmaps or
* compressed data.
*/
class BackendTextureData {
public:
enum class Type { kColor, kPixmaps, kCompressed };
BackendTextureData() = default;
BackendTextureData(const SkColor4f& color) : fType(Type::kColor), fColor(color) {}
BackendTextureData(const GrPixmap pixmaps[]) : fType(Type::kPixmaps), fPixmaps(pixmaps) {
SkASSERT(pixmaps);
}
BackendTextureData(const void* data, size_t size) : fType(Type::kCompressed) {
SkASSERT(data);
fCompressed.fData = data;
fCompressed.fSize = size;
}
Type type() const { return fType; }
SkColor4f color() const {
SkASSERT(this->type() == Type::kColor);
return fColor;
}
const GrPixmap& pixmap(int i) const {
SkASSERT(this->type() == Type::kPixmaps);
return fPixmaps[i];
}
const GrPixmap* pixmaps() const {
SkASSERT(this->type() == Type::kPixmaps);
return fPixmaps;
}
const void* compressedData() const {
SkASSERT(this->type() == Type::kCompressed);
return fCompressed.fData;
}
size_t compressedSize() const {
SkASSERT(this->type() == Type::kCompressed);
return fCompressed.fSize;
}
private:
Type fType = Type::kColor;
union {
SkColor4f fColor = {0, 0, 0, 0};
const GrPixmap* fPixmaps;
struct {
const void* fData;
size_t fSize;
} fCompressed;
};
};
/** /**
* Creates a texture directly in the backend API without wrapping it in a GrTexture. * Creates a texture directly in the backend API without wrapping it in a GrTexture.
@ -570,9 +518,9 @@ public:
GrMipmapped, GrMipmapped,
GrProtected); GrProtected);
bool updateBackendTexture(const GrBackendTexture&, bool clearBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*); std::array<float, 4> color);
/** /**
* Same as the createBackendTexture case except compressed backend textures can * Same as the createBackendTexture case except compressed backend textures can
@ -585,7 +533,8 @@ public:
bool updateCompressedBackendTexture(const GrBackendTexture&, bool updateCompressedBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*); const void* data,
size_t length);
virtual bool setBackendTextureState(const GrBackendTexture&, virtual bool setBackendTextureState(const GrBackendTexture&,
const GrBackendSurfaceMutableState&, const GrBackendSurfaceMutableState&,
@ -685,9 +634,11 @@ public:
virtual void xferBarrier(GrRenderTarget*, GrXferBarrierType) = 0; virtual void xferBarrier(GrRenderTarget*, GrXferBarrierType) = 0;
protected: protected:
static bool MipMapsAreCorrect(SkISize dimensions, GrMipmapped, const BackendTextureData*); static bool CompressedDataIsCorrect(SkISize dimensions,
static bool CompressedDataIsCorrect(SkISize dimensions, SkImage::CompressionType, SkImage::CompressionType,
GrMipmapped, const BackendTextureData*); GrMipmapped,
const void* data,
size_t length);
// Handles cases where a surface will be updated without a call to flushRenderTarget. // Handles cases where a surface will be updated without a call to flushRenderTarget.
void didWriteToSurface(GrSurface* surface, GrSurfaceOrigin origin, const SkIRect* bounds, void didWriteToSurface(GrSurface* surface, GrSurfaceOrigin origin, const SkIRect* bounds,
@ -710,13 +661,14 @@ private:
virtual GrBackendTexture onCreateCompressedBackendTexture( virtual GrBackendTexture onCreateCompressedBackendTexture(
SkISize dimensions, const GrBackendFormat&, GrMipmapped, GrProtected) = 0; SkISize dimensions, const GrBackendFormat&, GrMipmapped, GrProtected) = 0;
virtual bool onUpdateBackendTexture(const GrBackendTexture&, virtual bool onClearBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) = 0; std::array<float, 4> color) = 0;
virtual bool onUpdateCompressedBackendTexture(const GrBackendTexture&, virtual bool onUpdateCompressedBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) = 0; const void* data,
size_t length) = 0;
// called when the 3D context state is unknown. Subclass should emit any // called when the 3D context state is unknown. Subclass should emit any
// assumed 3D context state and dirty any state cache. // assumed 3D context state and dirty any state cache.

View File

@ -994,32 +994,12 @@ GrBackendTexture GrD3DGpu::onCreateBackendTexture(SkISize dimensions,
return GrBackendTexture(dimensions.width(), dimensions.height(), info); return GrBackendTexture(dimensions.width(), dimensions.height(), info);
} }
static void copy_src_data(char* mapPtr, static bool copy_color_data(const GrD3DCaps& caps,
DXGI_FORMAT dxgiFormat, char* mapPtr,
D3D12_PLACED_SUBRESOURCE_FOOTPRINT* placedFootprints, DXGI_FORMAT dxgiFormat,
const GrPixmap srcData[], SkISize dimensions,
int numMipLevels) {
SkASSERT(srcData && numMipLevels);
SkASSERT(!GrDxgiFormatIsCompressed(dxgiFormat));
SkASSERT(mapPtr);
size_t bytesPerPixel = GrDxgiFormatBytesPerBlock(dxgiFormat);
for (int currentMipLevel = 0; currentMipLevel < numMipLevels; currentMipLevel++) {
const size_t trimRowBytes = srcData[currentMipLevel].width() * bytesPerPixel;
// copy data into the buffer, skipping any trailing bytes
char* dst = mapPtr + placedFootprints[currentMipLevel].Offset;
SkRectMemcpy(dst, placedFootprints[currentMipLevel].Footprint.RowPitch,
srcData[currentMipLevel].addr(), srcData[currentMipLevel].rowBytes(),
trimRowBytes, srcData[currentMipLevel].height());
}
}
static bool copy_color_data(const GrD3DCaps& caps, char* mapPtr,
DXGI_FORMAT dxgiFormat, SkISize dimensions,
D3D12_PLACED_SUBRESOURCE_FOOTPRINT* placedFootprints, D3D12_PLACED_SUBRESOURCE_FOOTPRINT* placedFootprints,
SkColor4f color) { std::array<float, 4> color) {
auto colorType = caps.getFormatColorType(dxgiFormat); auto colorType = caps.getFormatColorType(dxgiFormat);
if (colorType == GrColorType::kUnknown) { if (colorType == GrColorType::kUnknown) {
return false; return false;
@ -1032,11 +1012,12 @@ static bool copy_color_data(const GrD3DCaps& caps, char* mapPtr,
return true; return true;
} }
bool GrD3DGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture, bool GrD3DGpu::onClearBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { std::array<float, 4> color) {
GrD3DTextureResourceInfo info; GrD3DTextureResourceInfo info;
SkAssertResult(backendTexture.getD3DTextureResourceInfo(&info)); SkAssertResult(backendTexture.getD3DTextureResourceInfo(&info));
SkASSERT(!GrDxgiFormatIsCompressed(info.fFormat));
sk_sp<GrD3DResourceState> state = backendTexture.getGrD3DResourceState(); sk_sp<GrD3DResourceState> state = backendTexture.getGrD3DResourceState();
SkASSERT(state); SkASSERT(state);
@ -1060,27 +1041,23 @@ bool GrD3DGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture,
D3D12_RESOURCE_DESC desc = d3dResource->GetDesc(); D3D12_RESOURCE_DESC desc = d3dResource->GetDesc();
unsigned int mipLevelCount = 1; unsigned int mipLevelCount = 1;
if (backendTexture.fMipmapped == GrMipmapped::kYes) { if (backendTexture.fMipmapped == GrMipmapped::kYes) {
mipLevelCount = SkMipmap::ComputeLevelCount(backendTexture.dimensions().width(), mipLevelCount = SkMipmap::ComputeLevelCount(backendTexture.dimensions()) + 1;
backendTexture.dimensions().height()) + 1;
} }
SkASSERT(mipLevelCount == info.fLevelCount); SkASSERT(mipLevelCount == info.fLevelCount);
SkAutoTMalloc<D3D12_PLACED_SUBRESOURCE_FOOTPRINT> placedFootprints(mipLevelCount); SkAutoSTMalloc<15, D3D12_PLACED_SUBRESOURCE_FOOTPRINT> placedFootprints(mipLevelCount);
UINT numRows;
UINT64 rowSizeInBytes;
UINT64 combinedBufferSize; UINT64 combinedBufferSize;
SkAutoTMalloc<UINT> numRows(mipLevelCount); // We reuse the same top-level buffer area for all levels, hence passing 1 for level count.
SkAutoTMalloc<UINT64> rowSizeInBytes(mipLevelCount); fDevice->GetCopyableFootprints(&desc,
fDevice->GetCopyableFootprints(&desc, 0, mipLevelCount, 0, placedFootprints.get(), /* first resource */ 0,
numRows.get(), rowSizeInBytes.get(), &combinedBufferSize); /* mip level count */ 1,
/* base offset */ 0,
placedFootprints.get(),
&numRows,
&rowSizeInBytes,
&combinedBufferSize);
SkASSERT(combinedBufferSize); SkASSERT(combinedBufferSize);
if (data->type() == BackendTextureData::Type::kColor &&
!GrDxgiFormatIsCompressed(info.fFormat) && mipLevelCount > 1) {
// For a single uncompressed color, we reuse the same top-level buffer area for all levels.
combinedBufferSize =
placedFootprints[0].Footprint.RowPitch * placedFootprints[0].Footprint.Height;
for (unsigned int i = 1; i < mipLevelCount; ++i) {
placedFootprints[i].Offset = 0;
placedFootprints[i].Footprint.RowPitch = placedFootprints[0].Footprint.RowPitch;
}
}
GrStagingBufferManager::Slice slice = fStagingBufferManager.allocateStagingBufferSlice( GrStagingBufferManager::Slice slice = fStagingBufferManager.allocateStagingBufferSlice(
combinedBufferSize, D3D12_TEXTURE_DATA_PLACEMENT_ALIGNMENT); combinedBufferSize, D3D12_TEXTURE_DATA_PLACEMENT_ALIGNMENT);
@ -1090,42 +1067,37 @@ bool GrD3DGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture,
char* bufferData = (char*)slice.fOffsetMapPtr; char* bufferData = (char*)slice.fOffsetMapPtr;
SkASSERT(bufferData); SkASSERT(bufferData);
if (!copy_color_data(this->d3dCaps(),
if (data->type() == BackendTextureData::Type::kPixmaps) { bufferData,
copy_src_data(bufferData, info.fFormat, placedFootprints.get(), data->pixmaps(), info.fFormat,
info.fLevelCount); backendTexture.dimensions(),
} else if (data->type() == BackendTextureData::Type::kCompressed) { placedFootprints,
copy_compressed_data(bufferData, info.fFormat, placedFootprints.get(), numRows.get(), color)) {
rowSizeInBytes.get(), data->compressedData(), info.fLevelCount); return false;
} else {
SkASSERT(data->type() == BackendTextureData::Type::kColor);
SkImage::CompressionType compression =
GrBackendFormatToCompressionType(backendTexture.getBackendFormat());
if (SkImage::CompressionType::kNone == compression) {
if (!copy_color_data(this->d3dCaps(), bufferData, info.fFormat,
backendTexture.dimensions(), placedFootprints, data->color())) {
return false;
}
} else {
size_t totalCompressedSize = SkCompressedFormatDataSize(compression,
backendTexture.dimensions(),
backendTexture.hasMipmaps());
SkAutoTMalloc<char> tempData(totalCompressedSize);
GrFillInCompressedData(compression, backendTexture.dimensions(),
backendTexture.fMipmapped, tempData, data->color());
copy_compressed_data(bufferData, info.fFormat, placedFootprints.get(), numRows.get(),
rowSizeInBytes.get(), tempData.get(), info.fLevelCount);
}
} }
// Update the offsets in the footprint to be relative to the slice's offset
// Update the offsets in the footprints to be relative to the slice's offset placedFootprints[0].Offset += slice.fOffset;
for (unsigned int i = 0; i < mipLevelCount; ++i) { // Since we're sharing data for all the levels, set all the upper level footprints to the base.
placedFootprints[i].Offset += slice.fOffset; UINT w = placedFootprints[0].Footprint.Width;
UINT h = placedFootprints[0].Footprint.Height;
for (unsigned int i = 1; i < mipLevelCount; ++i) {
w = std::max(1U, w/2);
h = std::max(1U, h/2);
placedFootprints[i].Offset = placedFootprints[0].Offset;
placedFootprints[i].Footprint.Format = placedFootprints[0].Footprint.Format;
placedFootprints[i].Footprint.Width = w;
placedFootprints[i].Footprint.Height = h;
placedFootprints[i].Footprint.Depth = 1;
placedFootprints[i].Footprint.RowPitch = placedFootprints[0].Footprint.RowPitch;
} }
ID3D12Resource* d3dBuffer = static_cast<GrD3DBuffer*>(slice.fBuffer)->d3dResource(); ID3D12Resource* d3dBuffer = static_cast<GrD3DBuffer*>(slice.fBuffer)->d3dResource();
cmdList->copyBufferToTexture(d3dBuffer, texture.get(), mipLevelCount, placedFootprints.get(), 0, cmdList->copyBufferToTexture(d3dBuffer,
0); texture.get(),
mipLevelCount,
placedFootprints.get(),
/*left*/ 0,
/*top */ 0);
if (finishedCallback) { if (finishedCallback) {
this->addFinishedCallback(std::move(finishedCallback)); this->addFinishedCallback(std::move(finishedCallback));
@ -1143,8 +1115,88 @@ GrBackendTexture GrD3DGpu::onCreateCompressedBackendTexture(
bool GrD3DGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTexture, bool GrD3DGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { const void* data,
return this->onUpdateBackendTexture(backendTexture, std::move(finishedCallback), data); size_t size) {
GrD3DTextureResourceInfo info;
SkAssertResult(backendTexture.getD3DTextureResourceInfo(&info));
sk_sp<GrD3DResourceState> state = backendTexture.getGrD3DResourceState();
SkASSERT(state);
sk_sp<GrD3DTexture> texture = GrD3DTexture::MakeWrappedTexture(this,
backendTexture.dimensions(),
GrWrapCacheable::kNo,
kRW_GrIOType,
info,
std::move(state));
if (!texture) {
return false;
}
GrD3DDirectCommandList* cmdList = this->currentCommandList();
if (!cmdList) {
return false;
}
texture->setResourceState(this, D3D12_RESOURCE_STATE_COPY_DEST);
ID3D12Resource* d3dResource = texture->d3dResource();
SkASSERT(d3dResource);
D3D12_RESOURCE_DESC desc = d3dResource->GetDesc();
unsigned int mipLevelCount = 1;
if (backendTexture.hasMipmaps()) {
mipLevelCount = SkMipmap::ComputeLevelCount(backendTexture.dimensions().width(),
backendTexture.dimensions().height()) + 1;
}
SkASSERT(mipLevelCount == info.fLevelCount);
SkAutoTMalloc<D3D12_PLACED_SUBRESOURCE_FOOTPRINT> placedFootprints(mipLevelCount);
UINT64 combinedBufferSize;
SkAutoTMalloc<UINT> numRows(mipLevelCount);
SkAutoTMalloc<UINT64> rowSizeInBytes(mipLevelCount);
fDevice->GetCopyableFootprints(&desc,
0,
mipLevelCount,
0,
placedFootprints.get(),
numRows.get(),
rowSizeInBytes.get(),
&combinedBufferSize);
SkASSERT(combinedBufferSize);
SkASSERT(!GrDxgiFormatIsCompressed(info.fFormat));
GrStagingBufferManager::Slice slice = fStagingBufferManager.allocateStagingBufferSlice(
combinedBufferSize, D3D12_TEXTURE_DATA_PLACEMENT_ALIGNMENT);
if (!slice.fBuffer) {
return false;
}
char* bufferData = (char*)slice.fOffsetMapPtr;
SkASSERT(bufferData);
copy_compressed_data(bufferData,
info.fFormat,
placedFootprints.get(),
numRows.get(),
rowSizeInBytes.get(),
data,
info.fLevelCount);
// Update the offsets in the footprints to be relative to the slice's offset
for (unsigned int i = 0; i < mipLevelCount; ++i) {
placedFootprints[i].Offset += slice.fOffset;
}
ID3D12Resource* d3dBuffer = static_cast<GrD3DBuffer*>(slice.fBuffer)->d3dResource();
cmdList->copyBufferToTexture(d3dBuffer,
texture.get(),
mipLevelCount,
placedFootprints.get(),
0,
0);
if (finishedCallback) {
this->addFinishedCallback(std::move(finishedCallback));
}
return true;
} }
void GrD3DGpu::deleteBackendTexture(const GrBackendTexture& tex) { void GrD3DGpu::deleteBackendTexture(const GrBackendTexture& tex) {

View File

@ -217,9 +217,9 @@ private:
GrMipmapped, GrMipmapped,
GrProtected) override; GrProtected) override;
bool onUpdateBackendTexture(const GrBackendTexture&, bool onClearBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; std::array<float, 4> color) override;
GrBackendTexture onCreateCompressedBackendTexture(SkISize dimensions, GrBackendTexture onCreateCompressedBackendTexture(SkISize dimensions,
const GrBackendFormat&, const GrBackendFormat&,
@ -228,7 +228,8 @@ private:
bool onUpdateCompressedBackendTexture(const GrBackendTexture&, bool onUpdateCompressedBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; const void* data,
size_t size) override;
bool submitDirectCommandList(SyncQueue sync); bool submitDirectCommandList(SyncQueue sync);

View File

@ -381,35 +381,23 @@ void GrDawnGpu::uploadTextureData(GrColorType srcColorType, const GrMipLevel tex
} }
} }
bool GrDawnGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture, bool GrDawnGpu::onClearBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { std::array<float, 4> color) {
GrDawnTextureInfo info; GrDawnTextureInfo info;
SkAssertResult(backendTexture.getDawnTextureInfo(&info)); SkAssertResult(backendTexture.getDawnTextureInfo(&info));
size_t bpp = GrDawnBytesPerBlock(info.fFormat);
size_t baseLayerSize = bpp * backendTexture.width() * backendTexture.height();
const void* pixels;
SkAutoMalloc defaultStorage(baseLayerSize);
if (data && data->type() == BackendTextureData::Type::kPixmaps) {
int numMipLevels = info.fLevelCount;
SkAutoTArray<GrMipLevel> texels(numMipLevels);
GrColorType colorType = data->pixmap(0).colorType();
for (int i = 0; i < numMipLevels; ++i) {
texels[i] = {data->pixmap(i).addr(), data->pixmap(i).rowBytes(), nullptr};
}
SkIRect dstRect = SkIRect::MakeSize(backendTexture.dimensions());
this->uploadTextureData(colorType, texels.get(), numMipLevels, dstRect, info.fTexture);
return true;
}
pixels = defaultStorage.get();
GrColorType colorType; GrColorType colorType;
if (!GrDawnFormatToGrColorType(info.fFormat, &colorType)) { if (!GrDawnFormatToGrColorType(info.fFormat, &colorType)) {
return false; return false;
} }
SkISize size{backendTexture.width(), backendTexture.height()};
GrImageInfo imageInfo(colorType, kUnpremul_SkAlphaType, nullptr, size); size_t bpp = GrDawnBytesPerBlock(info.fFormat);
GrClearImage(imageInfo, defaultStorage.get(), bpp * backendTexture.width(), data->color()); size_t baseLayerSize = bpp * backendTexture.width() * backendTexture.height();
SkAutoMalloc defaultStorage(baseLayerSize);
GrImageInfo imageInfo(colorType, kUnpremul_SkAlphaType, nullptr, backendTexture.dimensions());
GrClearImage(imageInfo, defaultStorage.get(), bpp * backendTexture.width(), color);
wgpu::Device device = this->device(); wgpu::Device device = this->device();
wgpu::CommandEncoder copyEncoder = this->getCopyEncoder(); wgpu::CommandEncoder copyEncoder = this->getCopyEncoder();
int w = backendTexture.width(), h = backendTexture.height(); int w = backendTexture.width(), h = backendTexture.height();
@ -420,9 +408,9 @@ bool GrDawnGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture,
GrStagingBufferManager::Slice stagingBuffer = GrStagingBufferManager::Slice stagingBuffer =
this->stagingBufferManager()->allocateStagingBufferSlice(size); this->stagingBufferManager()->allocateStagingBufferSlice(size);
if (rowBytes == origRowBytes) { if (rowBytes == origRowBytes) {
memcpy(stagingBuffer.fOffsetMapPtr, pixels, size); memcpy(stagingBuffer.fOffsetMapPtr, defaultStorage.get(), size);
} else { } else {
const char* src = static_cast<const char*>(pixels); const char* src = static_cast<const char*>(defaultStorage.get());
char* dst = static_cast<char*>(stagingBuffer.fOffsetMapPtr); char* dst = static_cast<char*>(stagingBuffer.fOffsetMapPtr);
for (int row = 0; row < h; row++) { for (int row = 0; row < h; row++) {
memcpy(dst, src, origRowBytes); memcpy(dst, src, origRowBytes);
@ -454,7 +442,8 @@ GrBackendTexture GrDawnGpu::onCreateCompressedBackendTexture(
bool GrDawnGpu::onUpdateCompressedBackendTexture(const GrBackendTexture&, bool GrDawnGpu::onUpdateCompressedBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) { const void* data,
size_t size) {
return false; return false;
} }

View File

@ -144,9 +144,9 @@ private:
GrMipmapped, GrMipmapped,
GrProtected) override; GrProtected) override;
bool onUpdateBackendTexture(const GrBackendTexture&, bool onClearBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; std::array<float, 4> color) override;
GrBackendTexture onCreateCompressedBackendTexture(SkISize dimensions, GrBackendTexture onCreateCompressedBackendTexture(SkISize dimensions,
const GrBackendFormat&, const GrBackendFormat&,
@ -155,7 +155,8 @@ private:
bool onUpdateCompressedBackendTexture(const GrBackendTexture&, bool onUpdateCompressedBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; const void* data,
size_t size) override;
sk_sp<GrGpuBuffer> onCreateBuffer(size_t size, GrGpuBufferType type, GrAccessPattern, sk_sp<GrGpuBuffer> onCreateBuffer(size_t size, GrGpuBufferType type, GrAccessPattern,
const void* data) override; const void* data) override;

View File

@ -980,7 +980,7 @@ bool GrGLGpu::uploadColorTypeTexData(GrGLFormat textureFormat,
bool GrGLGpu::uploadColorToTex(GrGLFormat textureFormat, bool GrGLGpu::uploadColorToTex(GrGLFormat textureFormat,
SkISize texDims, SkISize texDims,
GrGLenum target, GrGLenum target,
SkColor4f color, std::array<float, 4> color,
uint32_t levelMask) { uint32_t levelMask) {
GrColorType colorType; GrColorType colorType;
GrGLenum externalFormat, externalType; GrGLenum externalFormat, externalType;
@ -1386,8 +1386,11 @@ sk_sp<GrTexture> GrGLGpu::onCreateTexture(SkISize dimensions,
fHWBoundRenderTargetUniqueID.makeInvalid(); fHWBoundRenderTargetUniqueID.makeInvalid();
} else { } else {
this->bindTextureToScratchUnit(texDesc.fTarget, tex->textureID()); this->bindTextureToScratchUnit(texDesc.fTarget, tex->textureID());
static constexpr SkColor4f kZeroColor = {0, 0, 0, 0}; std::array<float, 4> zeros = {};
this->uploadColorToTex(texDesc.fFormat, texDesc.fSize, texDesc.fTarget, kZeroColor, this->uploadColorToTex(texDesc.fFormat,
texDesc.fSize,
texDesc.fTarget,
zeros,
levelClearMask); levelClearMask);
} }
} }
@ -1482,11 +1485,8 @@ GrBackendTexture GrGLGpu::onCreateCompressedBackendTexture(
bool GrGLGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTexture, bool GrGLGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { const void* data,
SkASSERT(data && data->type() != BackendTextureData::Type::kPixmaps); size_t length) {
this->handleDirtyContext();
GrGLTextureInfo info; GrGLTextureInfo info;
SkAssertResult(backendTexture.getGLTextureInfo(&info)); SkAssertResult(backendTexture.getGLTextureInfo(&info));
@ -1499,27 +1499,6 @@ bool GrGLGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTe
GrMipmapped mipMapped = backendTexture.hasMipmaps() ? GrMipmapped::kYes : GrMipmapped::kNo; GrMipmapped mipMapped = backendTexture.hasMipmaps() ? GrMipmapped::kYes : GrMipmapped::kNo;
const char* rawData = nullptr;
size_t rawDataSize = 0;
SkAutoMalloc am;
if (data->type() == BackendTextureData::Type::kCompressed) {
rawData = (const char*)data->compressedData();
rawDataSize = data->compressedSize();
} else {
SkASSERT(data->type() == BackendTextureData::Type::kColor);
SkASSERT(compression != SkImage::CompressionType::kNone);
rawDataSize = SkCompressedDataSize(compression, backendTexture.dimensions(), nullptr,
backendTexture.hasMipmaps());
am.reset(rawDataSize);
GrFillInCompressedData(compression, backendTexture.dimensions(), mipMapped, (char*)am.get(),
data->color());
rawData = (const char*)am.get();
}
this->bindTextureToScratchUnit(info.fTarget, info.fID); this->bindTextureToScratchUnit(info.fTarget, info.fID);
// If we have mips make sure the base level is set to 0 and the max level set to numMipLevels-1 // If we have mips make sure the base level is set to 0 and the max level set to numMipLevels-1
@ -1540,9 +1519,13 @@ bool GrGLGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTe
params->set(nullptr, nonsamplerState, fResetTimestampForTextureParameters); params->set(nullptr, nonsamplerState, fResetTimestampForTextureParameters);
} }
bool result = this->uploadCompressedTexData( bool result = this->uploadCompressedTexData(compression,
compression, glFormat, backendTexture.dimensions(), mipMapped, GR_GL_TEXTURE_2D, glFormat,
rawData, rawDataSize); backendTexture.dimensions(),
mipMapped,
GR_GL_TEXTURE_2D,
data,
length);
// Unbind this texture from the scratch texture unit. // Unbind this texture from the scratch texture unit.
this->bindTextureToScratchUnit(info.fTarget, 0); this->bindTextureToScratchUnit(info.fTarget, 0);
@ -3617,9 +3600,9 @@ GrBackendTexture GrGLGpu::onCreateBackendTexture(SkISize dimensions,
std::move(parameters)); std::move(parameters));
} }
bool GrGLGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture, bool GrGLGpu::onClearBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { std::array<float, 4> color) {
this->handleDirtyContext(); this->handleDirtyContext();
GrGLTextureInfo info; GrGLTextureInfo info;
@ -3651,26 +3634,12 @@ bool GrGLGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture,
params->set(nullptr, nonsamplerState, fResetTimestampForTextureParameters); params->set(nullptr, nonsamplerState, fResetTimestampForTextureParameters);
} }
SkASSERT(data->type() != BackendTextureData::Type::kCompressed); uint32_t levelMask = (1 << numMipLevels) - 1;
bool result = false; bool result = this->uploadColorToTex(glFormat,
if (data->type() == BackendTextureData::Type::kPixmaps) { backendTexture.dimensions(),
SkTArray<GrMipLevel> texels; info.fTarget,
texels.push_back_n(numMipLevels); color,
GrColorType colorType = data->pixmap(0).colorType(); levelMask);
for (int i = 0; i < numMipLevels; ++i) {
texels[i] = {data->pixmap(i).addr(),
data->pixmap(i).rowBytes(),
data->pixmap(i).pixelStorage()};
}
SkIRect dstRect = SkIRect::MakeSize(backendTexture.dimensions());
result = this->uploadColorTypeTexData(glFormat, colorType, backendTexture.dimensions(),
info.fTarget, dstRect, colorType, texels.begin(),
texels.count());
} else if (data->type() == BackendTextureData::Type::kColor) {
uint32_t levelMask = (1 << numMipLevels) - 1;
result = this->uploadColorToTex(glFormat, backendTexture.dimensions(), info.fTarget,
data->color(), levelMask);
}
// Unbind this texture from the scratch texture unit. // Unbind this texture from the scratch texture unit.
this->bindTextureToScratchUnit(info.fTarget, 0); this->bindTextureToScratchUnit(info.fTarget, 0);

View File

@ -212,13 +212,14 @@ private:
GrMipmapped, GrMipmapped,
GrProtected) override; GrProtected) override;
bool onUpdateBackendTexture(const GrBackendTexture&, bool onClearBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; std::array<float, 4> color) override;
bool onUpdateCompressedBackendTexture(const GrBackendTexture&, bool onUpdateCompressedBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; const void* data,
size_t length) override;
void onResetContext(uint32_t resetBits) override; void onResetContext(uint32_t resetBits) override;
@ -453,7 +454,7 @@ private:
bool uploadColorToTex(GrGLFormat textureFormat, bool uploadColorToTex(GrGLFormat textureFormat,
SkISize texDims, SkISize texDims,
GrGLenum target, GrGLenum target,
SkColor4f color, std::array<float, 4> color,
uint32_t levelMask); uint32_t levelMask);
// Pushes data to the currently bound texture to the currently active unit. 'dstRect' must be // Pushes data to the currently bound texture to the currently active unit. 'dstRect' must be

View File

@ -164,9 +164,9 @@ private:
GrMipmapped, GrMipmapped,
GrProtected) override; GrProtected) override;
bool onUpdateBackendTexture(const GrBackendTexture&, bool onClearBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override { std::array<float, 4> color) override {
return true; return true;
} }
@ -177,7 +177,8 @@ private:
bool onUpdateCompressedBackendTexture(const GrBackendTexture&, bool onUpdateCompressedBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override { const void*,
size_t) override {
return true; return true;
} }

View File

@ -131,9 +131,9 @@ private:
GrMipmapped, GrMipmapped,
GrProtected) override; GrProtected) override;
bool onUpdateBackendTexture(const GrBackendTexture&, bool onClearBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; std::array<float, 4> color) override;
GrBackendTexture onCreateCompressedBackendTexture(SkISize dimensions, GrBackendTexture onCreateCompressedBackendTexture(SkISize dimensions,
const GrBackendFormat&, const GrBackendFormat&,
@ -142,7 +142,8 @@ private:
bool onUpdateCompressedBackendTexture(const GrBackendTexture&, bool onUpdateCompressedBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; const void* data,
size_t size) override;
sk_sp<GrTexture> onCreateTexture(SkISize, sk_sp<GrTexture> onCreateTexture(SkISize,
const GrBackendFormat&, const GrBackendFormat&,

View File

@ -946,9 +946,9 @@ GrBackendTexture GrMtlGpu::onCreateBackendTexture(SkISize dimensions,
return backendTex; return backendTex;
} }
bool GrMtlGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture, bool GrMtlGpu::onClearBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { std::array<float, 4> color) {
GrMtlTextureInfo info; GrMtlTextureInfo info;
SkAssertResult(backendTexture.getMtlTextureInfo(&info)); SkAssertResult(backendTexture.getMtlTextureInfo(&info));
@ -956,45 +956,19 @@ bool GrMtlGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture,
const MTLPixelFormat mtlFormat = mtlTexture.pixelFormat; const MTLPixelFormat mtlFormat = mtlTexture.pixelFormat;
int numMipLevels = mtlTexture.mipmapLevelCount;
GrMipmapped mipMapped = numMipLevels > 1 ? GrMipmapped::kYes : GrMipmapped::kNo;
SkImage::CompressionType compression = GrBackendFormatToCompressionType(
backendTexture.getBackendFormat());
// Create a transfer buffer and fill with data. // Create a transfer buffer and fill with data.
size_t bytesPerPixel = GrMtlFormatBytesPerBlock(mtlFormat); size_t bytesPerPixel = GrMtlFormatBytesPerBlock(mtlFormat);
SkSTArray<16, size_t> individualMipOffsets;
size_t combinedBufferSize; size_t combinedBufferSize;
if (data->type() == BackendTextureData::Type::kColor && // Reuse the same buffer for all levels. Should be ok since we made the row bytes tight.
compression == SkImage::CompressionType::kNone) { combinedBufferSize = bytesPerPixel*backendTexture.width()*backendTexture.height();
combinedBufferSize = bytesPerPixel*backendTexture.width()*backendTexture.height();
// Reuse the same buffer for all levels. Should be ok since we made the row bytes tight.
individualMipOffsets.push_back_n(numMipLevels, (size_t)0);
} else if (compression == SkImage::CompressionType::kNone) {
combinedBufferSize = GrComputeTightCombinedBufferSize(bytesPerPixel,
backendTexture.dimensions(),
&individualMipOffsets,
numMipLevels);
} else {
combinedBufferSize = SkCompressedDataSize(compression, backendTexture.dimensions(),
&individualMipOffsets,
mipMapped == GrMipmapped::kYes);
}
SkASSERT(individualMipOffsets.count() == numMipLevels);
#ifdef SK_BUILD_FOR_MAC #ifdef SK_BUILD_FOR_MAC
static const size_t kMinAlignment = 4; static const size_t kMinAlignment = 4;
#else #else
static const size_t kMinAlignment = 1; static const size_t kMinAlignment = 1;
#endif #endif
size_t alignment; size_t alignment = std::max(bytesPerPixel, kMinAlignment);
if (data->type() == BackendTextureData::Type::kCompressed) {
alignment = std::max(SkCompressedBlockSize(compression), kMinAlignment);
} else {
alignment = std::max(bytesPerPixel, kMinAlignment);
}
GrStagingBufferManager::Slice slice = fStagingBufferManager.allocateStagingBufferSlice( GrStagingBufferManager::Slice slice = fStagingBufferManager.allocateStagingBufferSlice(
combinedBufferSize, alignment); combinedBufferSize, alignment);
if (!slice.fBuffer) { if (!slice.fBuffer) {
@ -1002,28 +976,15 @@ bool GrMtlGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture,
} }
char* buffer = (char*)slice.fOffsetMapPtr; char* buffer = (char*)slice.fOffsetMapPtr;
if (data->type() == BackendTextureData::Type::kPixmaps) { auto colorType = mtl_format_to_backend_tex_clear_colortype(mtlFormat);
copy_src_data(buffer, bytesPerPixel, individualMipOffsets, data->pixmaps(), if (colorType == GrColorType::kUnknown) {
numMipLevels, combinedBufferSize); return false;
} else if (data->type() == BackendTextureData::Type::kCompressed) { }
memcpy(buffer, data->compressedData(), data->compressedSize()); GrImageInfo ii(colorType, kUnpremul_SkAlphaType, nullptr, backendTexture.dimensions());
} else { auto rb = ii.minRowBytes();
SkASSERT(data->type() == BackendTextureData::Type::kColor); SkASSERT(rb == bytesPerPixel*backendTexture.width());
if (compression == SkImage::CompressionType::kNone) { if (!GrClearImage(ii, buffer, rb, color)) {
auto colorType = mtl_format_to_backend_tex_clear_colortype(mtlFormat); return false;
if (colorType == GrColorType::kUnknown) {
return false;
}
GrImageInfo ii(colorType, kUnpremul_SkAlphaType, nullptr, backendTexture.dimensions());
auto rb = ii.minRowBytes();
SkASSERT(rb == bytesPerPixel*backendTexture.width());
if (!GrClearImage(ii, buffer, rb, data->color())) {
return false;
}
} else {
GrFillInCompressedData(compression, backendTexture.dimensions(), mipMapped, buffer,
data->color());
}
} }
// Transfer buffer contents to texture // Transfer buffer contents to texture
@ -1034,32 +995,29 @@ bool GrMtlGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture,
GrMtlBuffer* mtlBuffer = static_cast<GrMtlBuffer*>(slice.fBuffer); GrMtlBuffer* mtlBuffer = static_cast<GrMtlBuffer*>(slice.fBuffer);
SkISize levelDimensions(backendTexture.dimensions()); SkISize levelDimensions(backendTexture.dimensions());
int numMipLevels = mtlTexture.mipmapLevelCount;
for (int currentMipLevel = 0; currentMipLevel < numMipLevels; currentMipLevel++) { for (int currentMipLevel = 0; currentMipLevel < numMipLevels; currentMipLevel++) {
size_t levelRowBytes; size_t levelRowBytes;
size_t levelSize; size_t levelSize;
if (compression == SkImage::CompressionType::kNone) { levelRowBytes = levelDimensions.width() * bytesPerPixel;
levelRowBytes = levelDimensions.width() * bytesPerPixel; levelSize = levelRowBytes * levelDimensions.height();
levelSize = levelRowBytes * levelDimensions.height();
} else {
levelRowBytes = GrCompressedRowBytes(compression, levelDimensions.width());
levelSize = SkCompressedDataSize(compression, levelDimensions, nullptr, false);
}
// TODO: can this all be done in one go? // TODO: can this all be done in one go?
[blitCmdEncoder copyFromBuffer: mtlBuffer->mtlBuffer() [blitCmdEncoder copyFromBuffer: mtlBuffer->mtlBuffer()
sourceOffset: slice.fOffset + individualMipOffsets[currentMipLevel] sourceOffset: slice.fOffset
sourceBytesPerRow: levelRowBytes sourceBytesPerRow: levelRowBytes
sourceBytesPerImage: levelSize sourceBytesPerImage: levelSize
sourceSize: MTLSizeMake(levelDimensions.width(), sourceSize: MTLSizeMake(levelDimensions.width(),
levelDimensions.height(), 1) levelDimensions.height(),
1)
toTexture: mtlTexture toTexture: mtlTexture
destinationSlice: 0 destinationSlice: 0
destinationLevel: currentMipLevel destinationLevel: currentMipLevel
destinationOrigin: origin]; destinationOrigin: origin];
levelDimensions = { std::max(1, levelDimensions.width() / 2), levelDimensions = {std::max(1, levelDimensions.width() / 2),
std::max(1, levelDimensions.height() / 2) }; std::max(1, levelDimensions.height() / 2)};
} }
#ifdef SK_BUILD_FOR_MAC #ifdef SK_BUILD_FOR_MAC
[mtlBuffer->mtlBuffer() didModifyRange: NSMakeRange(slice.fOffset, combinedBufferSize)]; [mtlBuffer->mtlBuffer() didModifyRange: NSMakeRange(slice.fOffset, combinedBufferSize)];
@ -1088,8 +1046,84 @@ GrBackendTexture GrMtlGpu::onCreateCompressedBackendTexture(
bool GrMtlGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTexture, bool GrMtlGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { const void* data,
return this->onUpdateBackendTexture(backendTexture, std::move(finishedCallback), data); size_t size) {
GrMtlTextureInfo info;
SkAssertResult(backendTexture.getMtlTextureInfo(&info));
id<MTLTexture> mtlTexture = GrGetMTLTexture(info.fTexture.get());
int numMipLevels = mtlTexture.mipmapLevelCount;
GrMipmapped mipMapped = numMipLevels > 1 ? GrMipmapped::kYes : GrMipmapped::kNo;
SkImage::CompressionType compression =
GrBackendFormatToCompressionType(backendTexture.getBackendFormat());
SkASSERT(compression != SkImage::CompressionType::kNone);
// Create a transfer buffer and fill with data.
SkSTArray<16, size_t> individualMipOffsets;
size_t combinedBufferSize;
combinedBufferSize = SkCompressedDataSize(compression,
backendTexture.dimensions(),
&individualMipOffsets,
mipMapped == GrMipmapped::kYes);
SkASSERT(individualMipOffsets.count() == numMipLevels);
#ifdef SK_BUILD_FOR_MAC
static const size_t kMinAlignment = 4;
#else
static const size_t kMinAlignment = 1;
#endif
size_t alignment = std::max(SkCompressedBlockSize(compression), kMinAlignment);
GrStagingBufferManager::Slice slice =
fStagingBufferManager.allocateStagingBufferSlice(combinedBufferSize, alignment);
if (!slice.fBuffer) {
return false;
}
char* buffer = (char*)slice.fOffsetMapPtr;
memcpy(buffer, data, size);
// Transfer buffer contents to texture
MTLOrigin origin = MTLOriginMake(0, 0, 0);
GrMtlCommandBuffer* cmdBuffer = this->commandBuffer();
id<MTLBlitCommandEncoder> blitCmdEncoder = cmdBuffer->getBlitCommandEncoder();
GrMtlBuffer* mtlBuffer = static_cast<GrMtlBuffer*>(slice.fBuffer);
SkISize levelDimensions(backendTexture.dimensions());
for (int currentMipLevel = 0; currentMipLevel < numMipLevels; currentMipLevel++) {
size_t levelRowBytes;
size_t levelSize;
levelRowBytes = GrCompressedRowBytes(compression, levelDimensions.width());
levelSize = SkCompressedDataSize(compression, levelDimensions, nullptr, false);
// TODO: can this all be done in one go?
[blitCmdEncoder copyFromBuffer: mtlBuffer->mtlBuffer()
sourceOffset: slice.fOffset + individualMipOffsets[currentMipLevel]
sourceBytesPerRow: levelRowBytes
sourceBytesPerImage: levelSize
sourceSize: MTLSizeMake(levelDimensions.width(),
levelDimensions.height(),
1)
toTexture: mtlTexture
destinationSlice: 0
destinationLevel: currentMipLevel
destinationOrigin: origin];
levelDimensions = {std::max(1, levelDimensions.width() / 2),
std::max(1, levelDimensions.height() / 2)};
}
#ifdef SK_BUILD_FOR_MAC
[mtlBuffer->mtlBuffer() didModifyRange:NSMakeRange(slice.fOffset, combinedBufferSize)];
#endif
if (finishedCallback) {
this->addFinishedCallback(std::move(finishedCallback));
}
return true;
} }
void GrMtlGpu::deleteBackendTexture(const GrBackendTexture& tex) { void GrMtlGpu::deleteBackendTexture(const GrBackendTexture& tex) {

View File

@ -728,14 +728,17 @@ bool GrVkGpu::uploadTexDataLinear(GrVkAttachment* texAttachment, int left, int t
// This fills in the 'regions' vector in preparation for copying a buffer to an image. // This fills in the 'regions' vector in preparation for copying a buffer to an image.
// 'individualMipOffsets' is filled in as a side-effect. // 'individualMipOffsets' is filled in as a side-effect.
static size_t fill_in_regions(GrStagingBufferManager* stagingBufferManager, static size_t fill_in_compressed_regions(GrStagingBufferManager* stagingBufferManager,
SkTArray<VkBufferImageCopy>* regions, SkTArray<VkBufferImageCopy>* regions,
SkTArray<size_t>* individualMipOffsets, SkTArray<size_t>* individualMipOffsets,
GrStagingBufferManager::Slice* slice, GrStagingBufferManager::Slice* slice,
SkImage::CompressionType compression, SkImage::CompressionType compression,
VkFormat vkFormat, SkISize dimensions, GrMipmapped mipMapped) { VkFormat vkFormat,
SkISize dimensions,
GrMipmapped mipmapped) {
SkASSERT(compression != SkImage::CompressionType::kNone);
int numMipLevels = 1; int numMipLevels = 1;
if (mipMapped == GrMipmapped::kYes) { if (mipmapped == GrMipmapped::kYes) {
numMipLevels = SkMipmap::ComputeLevelCount(dimensions.width(), dimensions.height()) + 1; numMipLevels = SkMipmap::ComputeLevelCount(dimensions.width(), dimensions.height()) + 1;
} }
@ -744,15 +747,10 @@ static size_t fill_in_regions(GrStagingBufferManager* stagingBufferManager,
size_t bytesPerBlock = GrVkFormatBytesPerBlock(vkFormat); size_t bytesPerBlock = GrVkFormatBytesPerBlock(vkFormat);
size_t combinedBufferSize; size_t bufferSize = SkCompressedDataSize(compression,
if (compression == SkImage::CompressionType::kNone) { dimensions,
combinedBufferSize = GrComputeTightCombinedBufferSize(bytesPerBlock, dimensions, individualMipOffsets,
individualMipOffsets, mipmapped == GrMipmapped::kYes);
numMipLevels);
} else {
combinedBufferSize = SkCompressedDataSize(compression, dimensions, individualMipOffsets,
mipMapped == GrMipmapped::kYes);
}
SkASSERT(individualMipOffsets->count() == numMipLevels); SkASSERT(individualMipOffsets->count() == numMipLevels);
// Get a staging buffer slice to hold our mip data. // Get a staging buffer slice to hold our mip data.
@ -763,7 +761,7 @@ static size_t fill_in_regions(GrStagingBufferManager* stagingBufferManager,
case 2: alignment *= 2; break; // alignment is a multiple of 2 but not 4. case 2: alignment *= 2; break; // alignment is a multiple of 2 but not 4.
default: alignment *= 4; break; // alignment is not a multiple of 2. default: alignment *= 4; break; // alignment is not a multiple of 2.
} }
*slice = stagingBufferManager->allocateStagingBufferSlice(combinedBufferSize, alignment); *slice = stagingBufferManager->allocateStagingBufferSlice(bufferSize, alignment);
if (!slice->fBuffer) { if (!slice->fBuffer) {
return 0; return 0;
} }
@ -784,7 +782,7 @@ static size_t fill_in_regions(GrStagingBufferManager* stagingBufferManager,
std::max(1, dimensions.height()/2)}; std::max(1, dimensions.height()/2)};
} }
return combinedBufferSize; return bufferSize;
} }
bool GrVkGpu::uploadTexDataOptimal(GrVkAttachment* texAttachment, int left, int top, int width, bool GrVkGpu::uploadTexDataOptimal(GrVkAttachment* texAttachment, int left, int top, int width,
@ -932,10 +930,14 @@ bool GrVkGpu::uploadTexDataCompressed(GrVkAttachment* uploadTexture,
GrStagingBufferManager::Slice slice; GrStagingBufferManager::Slice slice;
SkTArray<VkBufferImageCopy> regions; SkTArray<VkBufferImageCopy> regions;
SkTArray<size_t> individualMipOffsets; SkTArray<size_t> individualMipOffsets;
SkDEBUGCODE(size_t combinedBufferSize =) fill_in_regions(&fStagingBufferManager, SkDEBUGCODE(size_t combinedBufferSize =) fill_in_compressed_regions(&fStagingBufferManager,
&regions, &individualMipOffsets, &regions,
&slice, compression, vkFormat, &individualMipOffsets,
dimensions, mipMapped); &slice,
compression,
vkFormat,
dimensions,
mipMapped);
if (!slice.fBuffer) { if (!slice.fBuffer) {
return false; return false;
} }
@ -1461,22 +1463,6 @@ bool copy_src_data(char* mapPtr,
return true; return true;
} }
bool copy_compressed_data(GrVkGpu* gpu, char* mapPtr,
const void* rawData, size_t dataSize) {
SkASSERT(mapPtr);
memcpy(mapPtr, rawData, dataSize);
return true;
}
bool generate_compressed_data(GrVkGpu* gpu, char* mapPtr,
SkImage::CompressionType compression, SkISize dimensions,
GrMipmapped mipMapped, const SkColor4f& color) {
SkASSERT(mapPtr);
GrFillInCompressedData(compression, dimensions, mipMapped, mapPtr, color);
return true;
}
bool GrVkGpu::createVkImageForBackendSurface(VkFormat vkFormat, bool GrVkGpu::createVkImageForBackendSurface(VkFormat vkFormat,
SkISize dimensions, SkISize dimensions,
int sampleCnt, int sampleCnt,
@ -1544,9 +1530,9 @@ bool GrVkGpu::createVkImageForBackendSurface(VkFormat vkFormat,
return true; return true;
} }
bool GrVkGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture, bool GrVkGpu::onClearBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { std::array<float, 4> color) {
GrVkImageInfo info; GrVkImageInfo info;
SkAssertResult(backendTexture.getVkImageInfo(&info)); SkAssertResult(backendTexture.getVkImageInfo(&info));
@ -1570,68 +1556,23 @@ bool GrVkGpu::onUpdateBackendTexture(const GrBackendTexture& backendTexture,
VK_ACCESS_TRANSFER_WRITE_BIT, VK_PIPELINE_STAGE_TRANSFER_BIT, VK_ACCESS_TRANSFER_WRITE_BIT, VK_PIPELINE_STAGE_TRANSFER_BIT,
false); false);
// Unfortunately, CmdClearColorImage doesn't work for compressed formats // CmdClearColorImage doesn't work for compressed formats
bool fastPath = data->type() == BackendTextureData::Type::kColor && SkASSERT(!GrVkFormatIsCompressed(info.fFormat));
!GrVkFormatIsCompressed(info.fFormat);
if (fastPath) { VkClearColorValue vkColor;
SkASSERT(data->type() == BackendTextureData::Type::kColor); // If we ever support SINT or UINT formats this needs to be updated to use the int32 and
VkClearColorValue vkColor; // uint32 union members in those cases.
SkColor4f color = data->color(); vkColor.float32[0] = color[0];
// If we ever support SINT or UINT formats this needs to be updated to use the int32 and vkColor.float32[1] = color[1];
// uint32 union members in those cases. vkColor.float32[2] = color[2];
vkColor.float32[0] = color.fR; vkColor.float32[3] = color[3];
vkColor.float32[1] = color.fG; VkImageSubresourceRange range;
vkColor.float32[2] = color.fB; range.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
vkColor.float32[3] = color.fA; range.baseArrayLayer = 0;
VkImageSubresourceRange range; range.baseMipLevel = 0;
range.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT; range.layerCount = 1;
range.baseArrayLayer = 0; range.levelCount = info.fLevelCount;
range.baseMipLevel = 0; cmdBuffer->clearColorImage(this, texAttachment, &vkColor, 1, &range);
range.layerCount = 1;
range.levelCount = info.fLevelCount;
cmdBuffer->clearColorImage(this, texAttachment, &vkColor, 1, &range);
} else {
SkImage::CompressionType compression = GrBackendFormatToCompressionType(
backendTexture.getBackendFormat());
SkTArray<VkBufferImageCopy> regions;
SkTArray<size_t> individualMipOffsets;
GrStagingBufferManager::Slice slice;
fill_in_regions(&fStagingBufferManager, &regions, &individualMipOffsets,
&slice, compression, info.fFormat, backendTexture.dimensions(),
backendTexture.fMipmapped);
if (!slice.fBuffer) {
return false;
}
bool result;
if (data->type() == BackendTextureData::Type::kPixmaps) {
result = copy_src_data((char*)slice.fOffsetMapPtr, info.fFormat, individualMipOffsets,
data->pixmaps(), info.fLevelCount);
} else if (data->type() == BackendTextureData::Type::kCompressed) {
result = copy_compressed_data(this, (char*)slice.fOffsetMapPtr,
data->compressedData(), data->compressedSize());
} else {
SkASSERT(data->type() == BackendTextureData::Type::kColor);
result = generate_compressed_data(this, (char*)slice.fOffsetMapPtr, compression,
backendTexture.dimensions(),
backendTexture.fMipmapped, data->color());
}
cmdBuffer->addGrSurface(texture);
// Copy the buffer to the image. This call takes the raw VkBuffer instead of a GrGpuBuffer
// because we don't need the command buffer to ref the buffer here. The reason being is that
// the buffer is coming from the staging manager and the staging manager will make sure the
// command buffer has a ref on the buffer. This avoids having to add and remove a ref for
// every upload in the frame.
const GrVkBuffer* vkBuffer = static_cast<GrVkBuffer*>(slice.fBuffer);
cmdBuffer->copyBufferToImage(this, vkBuffer->vkBuffer(), texAttachment,
texAttachment->currentLayout(), regions.count(),
regions.begin());
}
// Change image layout to shader read since if we use this texture as a borrowed // Change image layout to shader read since if we use this texture as a borrowed
// texture within Ganesh we require that its layout be set to that // texture within Ganesh we require that its layout be set to that
@ -1688,8 +1629,82 @@ GrBackendTexture GrVkGpu::onCreateCompressedBackendTexture(
bool GrVkGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTexture, bool GrVkGpu::onUpdateCompressedBackendTexture(const GrBackendTexture& backendTexture,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData* data) { const void* data,
return this->onUpdateBackendTexture(backendTexture, std::move(finishedCallback), data); size_t size) {
GrVkImageInfo info;
SkAssertResult(backendTexture.getVkImageInfo(&info));
sk_sp<GrBackendSurfaceMutableStateImpl> mutableState = backendTexture.getMutableState();
SkASSERT(mutableState);
sk_sp<GrVkTexture> texture = GrVkTexture::MakeWrappedTexture(this,
backendTexture.dimensions(),
kBorrow_GrWrapOwnership,
GrWrapCacheable::kNo,
kRW_GrIOType,
info,
std::move(mutableState));
if (!texture) {
return false;
}
GrVkPrimaryCommandBuffer* cmdBuffer = this->currentCommandBuffer();
if (!cmdBuffer) {
return false;
}
GrVkAttachment* attachment = texture->textureAttachment();
attachment->setImageLayout(this,
VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
VK_ACCESS_TRANSFER_WRITE_BIT,
VK_PIPELINE_STAGE_TRANSFER_BIT,
false);
SkImage::CompressionType compression =
GrBackendFormatToCompressionType(backendTexture.getBackendFormat());
SkTArray<VkBufferImageCopy> regions;
SkTArray<size_t> individualMipOffsets;
GrStagingBufferManager::Slice slice;
fill_in_compressed_regions(&fStagingBufferManager,
&regions,
&individualMipOffsets,
&slice,
compression,
info.fFormat,
backendTexture.dimensions(),
backendTexture.fMipmapped);
if (!slice.fBuffer) {
return false;
}
memcpy(slice.fOffsetMapPtr, data, size);
cmdBuffer->addGrSurface(texture);
// Copy the buffer to the image. This call takes the raw VkBuffer instead of a GrGpuBuffer
// because we don't need the command buffer to ref the buffer here. The reason being is that
// the buffer is coming from the staging manager and the staging manager will make sure the
// command buffer has a ref on the buffer. This avoids having to add and remove a ref for
// every upload in the frame.
cmdBuffer->copyBufferToImage(this,
static_cast<GrVkBuffer*>(slice.fBuffer)->vkBuffer(),
attachment,
attachment->currentLayout(),
regions.count(),
regions.begin());
// Change image layout to shader read since if we use this texture as a borrowed
// texture within Ganesh we require that its layout be set to that
attachment->setImageLayout(this,
VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL,
VK_ACCESS_SHADER_READ_BIT,
VK_PIPELINE_STAGE_FRAGMENT_SHADER_BIT,
false);
if (finishedCallback) {
this->addFinishedCallback(std::move(finishedCallback));
}
return true;
} }
void set_layout_and_queue_from_mutable_state(GrVkGpu* gpu, GrVkImage* image, void set_layout_and_queue_from_mutable_state(GrVkGpu* gpu, GrVkImage* image,

View File

@ -218,13 +218,14 @@ private:
GrMipmapped, GrMipmapped,
GrProtected) override; GrProtected) override;
bool onUpdateBackendTexture(const GrBackendTexture&, bool onClearBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; std::array<float, 4> color) override;
bool onUpdateCompressedBackendTexture(const GrBackendTexture&, bool onUpdateCompressedBackendTexture(const GrBackendTexture&,
sk_sp<GrRefCntedCallback> finishedCallback, sk_sp<GrRefCntedCallback> finishedCallback,
const BackendTextureData*) override; const void* data,
size_t length) override;
bool setBackendSurfaceState(GrVkImageInfo info, bool setBackendSurfaceState(GrVkImageInfo info,
sk_sp<GrBackendSurfaceMutableStateImpl> currentState, sk_sp<GrBackendSurfaceMutableStateImpl> currentState,

View File

@ -222,7 +222,7 @@ static void check_solid_pixmap(skiatest::Reporter* reporter,
static SkColor4f get_expected_color(SkColor4f orig, GrColorType ct) { static SkColor4f get_expected_color(SkColor4f orig, GrColorType ct) {
GrImageInfo ii(ct, kUnpremul_SkAlphaType, nullptr, {1, 1}); GrImageInfo ii(ct, kUnpremul_SkAlphaType, nullptr, {1, 1});
std::unique_ptr<char[]> data(new char[ii.minRowBytes()]); std::unique_ptr<char[]> data(new char[ii.minRowBytes()]);
GrClearImage(ii, data.get(), ii.minRowBytes(), orig); GrClearImage(ii, data.get(), ii.minRowBytes(), orig.array());
// Read back to SkColor4f. // Read back to SkColor4f.
SkColor4f result; SkColor4f result;