blob: 3bb666e427d549553bab0422b07879413c5d89cb [file] [log] [blame]
/*
* Copyright 2012 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "GrSurface.h"
#include "GrContext.h"
#include "GrOpList.h"
#include "GrSurfacePriv.h"
#include "SkBitmap.h"
#include "SkGrPriv.h"
#include "SkImageEncoder.h"
#include "SkMathPriv.h"
#include <stdio.h>
GrSurface::~GrSurface() {
if (fLastOpList) {
fLastOpList->clearTarget();
}
SkSafeUnref(fLastOpList);
// check that invokeReleaseProc has been called (if needed)
SkASSERT(NULL == fReleaseProc);
}
size_t GrSurface::WorstCaseSize(const GrSurfaceDesc& desc, bool useNextPow2) {
size_t size;
int width = useNextPow2 ? GrNextPow2(desc.fWidth) : desc.fWidth;
int height = useNextPow2 ? GrNextPow2(desc.fHeight) : desc.fHeight;
bool isRenderTarget = SkToBool(desc.fFlags & kRenderTarget_GrSurfaceFlag);
if (isRenderTarget) {
// We own one color value for each MSAA sample.
int colorValuesPerPixel = SkTMax(1, desc.fSampleCnt);
if (desc.fSampleCnt) {
// Worse case, we own the resolve buffer so that is one more sample per pixel.
colorValuesPerPixel += 1;
}
SkASSERT(kUnknown_GrPixelConfig != desc.fConfig);
SkASSERT(!GrPixelConfigIsCompressed(desc.fConfig));
size_t colorBytes = (size_t) width * height * GrBytesPerPixel(desc.fConfig);
// This would be a nice assert to have (i.e., we aren't creating 0 width/height surfaces).
// Unfortunately Chromium seems to want to do this.
//SkASSERT(colorBytes > 0);
size = colorValuesPerPixel * colorBytes;
size += colorBytes/3; // in case we have to mipmap
} else {
if (GrPixelConfigIsCompressed(desc.fConfig)) {
size = GrCompressedFormatDataSize(desc.fConfig, width, height);
} else {
size = (size_t) width * height * GrBytesPerPixel(desc.fConfig);
}
size += size/3; // in case we have to mipmap
}
return size;
}
size_t GrSurface::ComputeSize(const GrSurfaceDesc& desc,
int colorSamplesPerPixel,
bool hasMIPMaps,
bool useNextPow2) {
size_t colorSize;
int width = useNextPow2 ? GrNextPow2(desc.fWidth) : desc.fWidth;
int height = useNextPow2 ? GrNextPow2(desc.fHeight) : desc.fHeight;
SkASSERT(kUnknown_GrPixelConfig != desc.fConfig);
if (GrPixelConfigIsCompressed(desc.fConfig)) {
colorSize = GrCompressedFormatDataSize(desc.fConfig, width, height);
} else {
colorSize = (size_t) width * height * GrBytesPerPixel(desc.fConfig);
}
SkASSERT(colorSize > 0);
size_t finalSize = colorSamplesPerPixel * colorSize;
if (hasMIPMaps) {
// We don't have to worry about the mipmaps being a different size than
// we'd expect because we never change fDesc.fWidth/fHeight.
finalSize += colorSize/3;
}
SkASSERT(finalSize <= WorstCaseSize(desc, useNextPow2));
return finalSize;
}
template<typename T> static bool adjust_params(int surfaceWidth,
int surfaceHeight,
size_t bpp,
int* left, int* top, int* width, int* height,
T** data,
size_t* rowBytes) {
if (!*rowBytes) {
*rowBytes = *width * bpp;
}
SkIRect subRect = SkIRect::MakeXYWH(*left, *top, *width, *height);
SkIRect bounds = SkIRect::MakeWH(surfaceWidth, surfaceHeight);
if (!subRect.intersect(bounds)) {
return false;
}
*data = reinterpret_cast<void*>(reinterpret_cast<intptr_t>(*data) +
(subRect.fTop - *top) * *rowBytes + (subRect.fLeft - *left) * bpp);
*left = subRect.fLeft;
*top = subRect.fTop;
*width = subRect.width();
*height = subRect.height();
return true;
}
bool GrSurfacePriv::AdjustReadPixelParams(int surfaceWidth,
int surfaceHeight,
size_t bpp,
int* left, int* top, int* width, int* height,
void** data,
size_t* rowBytes) {
return adjust_params<void>(surfaceWidth, surfaceHeight, bpp, left, top, width, height, data,
rowBytes);
}
bool GrSurfacePriv::AdjustWritePixelParams(int surfaceWidth,
int surfaceHeight,
size_t bpp,
int* left, int* top, int* width, int* height,
const void** data,
size_t* rowBytes) {
return adjust_params<const void>(surfaceWidth, surfaceHeight, bpp, left, top, width, height,
data, rowBytes);
}
//////////////////////////////////////////////////////////////////////////////
bool GrSurface::writePixels(int left, int top, int width, int height,
GrPixelConfig config, const void* buffer, size_t rowBytes,
uint32_t pixelOpsFlags) {
// go through context so that all necessary flushing occurs
GrContext* context = this->getContext();
if (nullptr == context) {
return false;
}
return context->writeSurfacePixels(this, left, top, width, height, config, buffer,
rowBytes, pixelOpsFlags);
}
bool GrSurface::readPixels(int left, int top, int width, int height,
GrPixelConfig config, void* buffer, size_t rowBytes,
uint32_t pixelOpsFlags) {
// go through context so that all necessary flushing occurs
GrContext* context = this->getContext();
if (nullptr == context) {
return false;
}
return context->readSurfacePixels(this, left, top, width, height, config, buffer,
rowBytes, pixelOpsFlags);
}
static bool encode_image_to_file(const char* path, const SkBitmap& src) {
SkFILEWStream file(path);
return file.isValid() && SkEncodeImage(&file, src, SkEncodedImageFormat::kPNG, 100);
}
// TODO: This should probably be a non-member helper function. It might only be needed in
// debug or developer builds.
bool GrSurface::savePixels(const char* filename) {
SkBitmap bm;
if (!bm.tryAllocPixels(SkImageInfo::MakeN32Premul(this->width(), this->height()))) {
return false;
}
bool result = this->readPixels(0, 0, this->width(), this->height(), kSkia8888_GrPixelConfig,
bm.getPixels());
if (!result) {
SkDebugf("------ failed to read pixels for %s\n", filename);
return false;
}
// remove any previous version of this file
remove(filename);
if (!encode_image_to_file(filename, bm)) {
SkDebugf("------ failed to encode %s\n", filename);
remove(filename); // remove any partial file
return false;
}
return true;
}
void GrSurface::flushWrites() {
if (!this->wasDestroyed()) {
this->getContext()->flushSurfaceWrites(this);
}
}
bool GrSurface::hasPendingRead() const {
const GrTexture* thisTex = this->asTexture();
if (thisTex && thisTex->internalHasPendingRead()) {
return true;
}
const GrRenderTarget* thisRT = this->asRenderTarget();
if (thisRT && thisRT->internalHasPendingRead()) {
return true;
}
return false;
}
bool GrSurface::hasPendingWrite() const {
const GrTexture* thisTex = this->asTexture();
if (thisTex && thisTex->internalHasPendingWrite()) {
return true;
}
const GrRenderTarget* thisRT = this->asRenderTarget();
if (thisRT && thisRT->internalHasPendingWrite()) {
return true;
}
return false;
}
bool GrSurface::hasPendingIO() const {
const GrTexture* thisTex = this->asTexture();
if (thisTex && thisTex->internalHasPendingIO()) {
return true;
}
const GrRenderTarget* thisRT = this->asRenderTarget();
if (thisRT && thisRT->internalHasPendingIO()) {
return true;
}
return false;
}
void GrSurface::onRelease() {
this->invokeReleaseProc();
this->INHERITED::onRelease();
}
void GrSurface::onAbandon() {
this->invokeReleaseProc();
this->INHERITED::onAbandon();
}
void GrSurface::setLastOpList(GrOpList* opList) {
if (fLastOpList) {
// The non-MDB world never closes so we can't check this condition
#ifdef ENABLE_MDB
SkASSERT(fLastOpList->isClosed());
#endif
fLastOpList->clearTarget();
}
SkRefCnt_SafeAssign(fLastOpList, opList);
}