2015-12-10 15:52:45 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2015 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2017-02-15 15:20:30 +00:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <sstream>
|
2017-02-13 16:39:43 +00:00
|
|
|
#include <string>
|
2015-12-10 15:52:45 +00:00
|
|
|
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "src/core/SkAutoPixmapStorage.h"
|
2020-07-14 21:16:32 +00:00
|
|
|
#include "src/core/SkMipmap.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "src/core/SkUtils.h"
|
|
|
|
#include "tools/flags/CommandLineFlags.h"
|
2017-04-19 18:05:14 +00:00
|
|
|
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "tools/fiddle/fiddle_main.h"
|
2015-12-10 15:52:45 +00:00
|
|
|
|
2019-03-21 16:31:36 +00:00
|
|
|
static DEFINE_double(duration, 1.0,
|
|
|
|
"The total duration, in seconds, of the animation we are drawing.");
|
|
|
|
static DEFINE_double(frame, 1.0,
|
|
|
|
"A double value in [0, 1] that specifies the point in animation to draw.");
|
2017-04-19 18:05:14 +00:00
|
|
|
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/gpu/GrBackendSurface.h"
|
2020-10-14 15:23:11 +00:00
|
|
|
#include "src/gpu/GrDirectContextPriv.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "src/gpu/GrGpu.h"
|
2019-08-14 21:00:30 +00:00
|
|
|
#include "src/gpu/GrRenderTarget.h"
|
2021-07-01 14:27:44 +00:00
|
|
|
#include "src/gpu/GrResourceProvider.h"
|
2020-12-08 16:51:02 +00:00
|
|
|
#include "tools/gpu/ManagedBackendTexture.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "tools/gpu/gl/GLTestContext.h"
|
2017-11-16 19:59:48 +00:00
|
|
|
|
2015-12-10 15:52:45 +00:00
|
|
|
// Globals externed in fiddle_main.h
|
2020-12-08 16:51:02 +00:00
|
|
|
sk_sp<sk_gpu_test::ManagedBackendTexture> backEndTexture;
|
|
|
|
sk_sp<sk_gpu_test::ManagedBackendTexture> backEndTextureRenderTarget;
|
2017-11-16 19:59:48 +00:00
|
|
|
|
|
|
|
sk_sp<GrRenderTarget> backingRenderTarget; // not externed
|
|
|
|
GrBackendRenderTarget backEndRenderTarget;
|
|
|
|
|
2015-12-10 15:52:45 +00:00
|
|
|
SkBitmap source;
|
2016-04-18 15:17:56 +00:00
|
|
|
sk_sp<SkImage> image;
|
2017-04-19 18:05:14 +00:00
|
|
|
double duration; // The total duration of the animation in seconds.
|
|
|
|
double frame; // A value in [0, 1] of where we are in the animation.
|
2015-12-10 15:52:45 +00:00
|
|
|
|
2017-02-15 15:20:30 +00:00
|
|
|
// Global used by the local impl of SkDebugf.
|
|
|
|
std::ostringstream gTextOutput;
|
2017-02-13 16:39:43 +00:00
|
|
|
|
2020-07-13 20:13:31 +00:00
|
|
|
// Global to record the GL driver info via create_direct_context().
|
2017-06-01 17:24:11 +00:00
|
|
|
std::ostringstream gGLDriverInfo;
|
|
|
|
|
2017-02-13 16:39:43 +00:00
|
|
|
void SkDebugf(const char * fmt, ...) {
|
|
|
|
va_list args;
|
|
|
|
va_start(args, fmt);
|
2017-02-15 15:20:30 +00:00
|
|
|
char formatbuffer[1024];
|
|
|
|
int n = vsnprintf(formatbuffer, sizeof(formatbuffer), fmt, args);
|
2017-02-13 16:39:43 +00:00
|
|
|
va_end(args);
|
|
|
|
if (n>=0 && n<=int(sizeof(formatbuffer))) {
|
2017-02-15 15:20:30 +00:00
|
|
|
gTextOutput.write(formatbuffer, n);
|
2017-02-13 16:39:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-10 15:52:45 +00:00
|
|
|
static void encode_to_base64(const void* data, size_t size, FILE* out) {
|
|
|
|
const uint8_t* input = reinterpret_cast<const uint8_t*>(data);
|
|
|
|
const uint8_t* end = &input[size];
|
|
|
|
static const char codes[] =
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|
|
|
"abcdefghijklmnopqrstuvwxyz0123456789+/";
|
|
|
|
while (input != end) {
|
|
|
|
uint8_t b = (*input & 0xFC) >> 2;
|
|
|
|
fputc(codes[b], out);
|
|
|
|
b = (*input & 0x03) << 4;
|
|
|
|
++input;
|
|
|
|
if (input == end) {
|
|
|
|
fputc(codes[b], out);
|
|
|
|
fputs("==", out);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
b |= (*input & 0xF0) >> 4;
|
|
|
|
fputc(codes[b], out);
|
|
|
|
b = (*input & 0x0F) << 2;
|
|
|
|
++input;
|
|
|
|
if (input == end) {
|
|
|
|
fputc(codes[b], out);
|
|
|
|
fputc('=', out);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
b |= (*input & 0xC0) >> 6;
|
|
|
|
fputc(codes[b], out);
|
|
|
|
b = *input & 0x3F;
|
|
|
|
fputc(codes[b], out);
|
|
|
|
++input;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-15 15:20:30 +00:00
|
|
|
static void dump_output(const void* data, size_t size,
|
2017-02-13 16:39:43 +00:00
|
|
|
const char* name, bool last = true) {
|
|
|
|
printf("\t\"%s\": \"", name);
|
2017-02-15 15:20:30 +00:00
|
|
|
encode_to_base64(data, size, stdout);
|
2017-02-13 16:39:43 +00:00
|
|
|
fputs(last ? "\"\n" : "\",\n", stdout);
|
|
|
|
}
|
|
|
|
|
2017-02-15 15:20:30 +00:00
|
|
|
static void dump_output(const sk_sp<SkData>& data,
|
|
|
|
const char* name, bool last = true) {
|
|
|
|
if (data) {
|
|
|
|
dump_output(data->data(), data->size(), name, last);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-11 20:03:13 +00:00
|
|
|
static sk_sp<SkData> encode_snapshot(const sk_sp<SkSurface>& surface) {
|
2016-04-18 15:17:56 +00:00
|
|
|
sk_sp<SkImage> img(surface->makeImageSnapshot());
|
2017-07-11 20:03:13 +00:00
|
|
|
return img ? img->encodeToData() : nullptr;
|
2015-12-10 15:52:45 +00:00
|
|
|
}
|
|
|
|
|
2017-06-14 19:54:32 +00:00
|
|
|
static SkCanvas* prepare_canvas(SkCanvas * canvas) {
|
|
|
|
canvas->clear(SK_ColorWHITE);
|
|
|
|
return canvas;
|
|
|
|
}
|
|
|
|
|
2020-03-19 19:54:28 +00:00
|
|
|
#ifdef SK_GL
|
2020-12-08 16:51:02 +00:00
|
|
|
static bool setup_backend_objects(GrDirectContext* dContext,
|
2017-11-17 19:59:43 +00:00
|
|
|
const SkBitmap& bm,
|
|
|
|
const DrawOptions& options) {
|
2020-12-08 16:51:02 +00:00
|
|
|
if (!dContext) {
|
2020-02-10 15:41:56 +00:00
|
|
|
fputs("Context is null.\n", stderr);
|
2017-11-16 19:59:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-12-19 18:15:02 +00:00
|
|
|
// This config must match the SkColorType used in draw.cpp in the SkImage and Surface factories
|
2020-12-08 16:51:02 +00:00
|
|
|
GrBackendFormat renderableFormat = dContext->defaultBackendFormat(kRGBA_8888_SkColorType,
|
|
|
|
GrRenderable::kYes);
|
2017-11-16 19:59:48 +00:00
|
|
|
|
|
|
|
if (!bm.empty()) {
|
2017-12-12 21:24:41 +00:00
|
|
|
SkPixmap originalPixmap;
|
|
|
|
SkPixmap* pixmap = &originalPixmap;
|
|
|
|
if (!bm.peekPixels(&originalPixmap)) {
|
2020-02-10 15:41:56 +00:00
|
|
|
fputs("Unable to peekPixels.\n", stderr);
|
2017-12-12 21:24:41 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkAutoPixmapStorage rgbaPixmap;
|
|
|
|
if (kN32_SkColorType != kRGBA_8888_SkColorType) {
|
|
|
|
if (!rgbaPixmap.tryAlloc(bm.info().makeColorType(kRGBA_8888_SkColorType))) {
|
2020-02-10 15:41:56 +00:00
|
|
|
fputs("Unable to alloc rgbaPixmap.\n", stderr);
|
2017-12-12 21:24:41 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!bm.readPixels(rgbaPixmap)) {
|
2020-02-10 15:41:56 +00:00
|
|
|
fputs("Unable to read rgbaPixmap.\n", stderr);
|
2017-12-12 21:24:41 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
pixmap = &rgbaPixmap;
|
|
|
|
}
|
2017-11-16 19:59:48 +00:00
|
|
|
|
2020-12-08 16:51:02 +00:00
|
|
|
backEndTexture = sk_gpu_test::ManagedBackendTexture::MakeFromPixmap(dContext,
|
|
|
|
*pixmap,
|
|
|
|
options.fMipMapping,
|
|
|
|
GrRenderable::kNo,
|
|
|
|
GrProtected::kNo);
|
|
|
|
if (!backEndTexture) {
|
|
|
|
fputs("Failed to create backEndTexture.\n", stderr);
|
2017-11-16 19:59:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2020-12-08 16:51:02 +00:00
|
|
|
auto resourceProvider = dContext->priv().resourceProvider();
|
|
|
|
|
|
|
|
SkISize offscreenDims = {options.fOffScreenWidth, options.fOffScreenHeight};
|
|
|
|
SkAutoTMalloc<uint32_t> data(offscreenDims.area());
|
|
|
|
sk_memset32(data.get(), 0, offscreenDims.area());
|
|
|
|
|
2017-11-17 19:59:43 +00:00
|
|
|
// This backend object should be renderable but not textureable. Given the limitations
|
|
|
|
// of how we're creating it though it will wind up being secretly textureable.
|
|
|
|
// We use this fact to initialize it with data but don't allow mipmaps
|
2021-04-02 18:36:58 +00:00
|
|
|
GrMipLevel level0 = {data.get(), offscreenDims.width()*sizeof(uint32_t), nullptr};
|
2017-11-17 19:59:43 +00:00
|
|
|
|
2020-12-08 16:51:02 +00:00
|
|
|
constexpr int kSampleCnt = 0;
|
2019-07-19 18:24:36 +00:00
|
|
|
sk_sp<GrTexture> tmp = resourceProvider->createTexture(
|
2021-08-13 20:20:18 +00:00
|
|
|
offscreenDims, renderableFormat, GrTextureType::k2D, GrColorType::kRGBA_8888,
|
|
|
|
GrRenderable::kYes, kSampleCnt, SkBudgeted::kNo, GrMipMapped::kNo, GrProtected::kNo,
|
|
|
|
&level0);
|
2017-11-16 19:59:48 +00:00
|
|
|
if (!tmp || !tmp->asRenderTarget()) {
|
2020-02-10 15:41:56 +00:00
|
|
|
fputs("GrTexture is invalid.\n", stderr);
|
2017-11-16 19:59:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
backingRenderTarget = sk_ref_sp(tmp->asRenderTarget());
|
|
|
|
|
2017-12-13 20:00:45 +00:00
|
|
|
backEndRenderTarget = backingRenderTarget->getBackendRenderTarget();
|
2017-11-16 19:59:48 +00:00
|
|
|
if (!backEndRenderTarget.isValid()) {
|
2020-02-10 15:41:56 +00:00
|
|
|
fputs("BackEndRenderTarget is invalid.\n", stderr);
|
2017-11-16 19:59:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2020-12-08 16:51:02 +00:00
|
|
|
backEndTextureRenderTarget = sk_gpu_test::ManagedBackendTexture::MakeWithData(
|
|
|
|
dContext,
|
|
|
|
options.fOffScreenWidth,
|
|
|
|
options.fOffScreenHeight,
|
|
|
|
renderableFormat,
|
|
|
|
SkColors::kTransparent,
|
|
|
|
options.fOffScreenMipMapping,
|
|
|
|
GrRenderable::kYes,
|
|
|
|
GrProtected::kNo);
|
|
|
|
if (!backEndTextureRenderTarget) {
|
|
|
|
fputs("Failed to create backendTextureRenderTarget.\n", stderr);
|
2017-11-16 19:59:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2020-03-19 19:54:28 +00:00
|
|
|
#endif
|
2017-11-16 19:59:48 +00:00
|
|
|
|
2017-04-19 18:05:14 +00:00
|
|
|
int main(int argc, char** argv) {
|
2019-03-20 15:50:33 +00:00
|
|
|
CommandLineFlags::Parse(argc, argv);
|
2017-04-19 18:05:14 +00:00
|
|
|
duration = FLAGS_duration;
|
|
|
|
frame = FLAGS_frame;
|
2017-02-13 16:39:43 +00:00
|
|
|
DrawOptions options = GetDrawOptions();
|
|
|
|
// If textOnly then only do one type of image, otherwise the text
|
|
|
|
// output is duplicated for each type.
|
|
|
|
if (options.textOnly) {
|
|
|
|
options.raster = true;
|
|
|
|
options.gpu = false;
|
|
|
|
options.pdf = false;
|
|
|
|
options.skp = false;
|
|
|
|
}
|
2015-12-10 15:52:45 +00:00
|
|
|
if (options.source) {
|
2016-08-02 21:40:46 +00:00
|
|
|
sk_sp<SkData> data(SkData::MakeFromFileName(options.source));
|
2015-12-10 15:52:45 +00:00
|
|
|
if (!data) {
|
|
|
|
perror(options.source);
|
|
|
|
return 1;
|
|
|
|
} else {
|
2016-04-18 15:17:56 +00:00
|
|
|
image = SkImage::MakeFromEncoded(std::move(data));
|
2015-12-10 15:52:45 +00:00
|
|
|
if (!image) {
|
|
|
|
perror("Unable to decode the source image.");
|
|
|
|
return 1;
|
|
|
|
}
|
2018-02-07 20:51:00 +00:00
|
|
|
SkAssertResult(image->asLegacyBitmap(&source));
|
2015-12-10 15:52:45 +00:00
|
|
|
}
|
|
|
|
}
|
2016-03-25 18:29:34 +00:00
|
|
|
sk_sp<SkData> rasterData, gpuData, pdfData, skpData;
|
2016-11-29 22:14:58 +00:00
|
|
|
SkColorType colorType = kN32_SkColorType;
|
|
|
|
sk_sp<SkColorSpace> colorSpace = nullptr;
|
|
|
|
if (options.f16) {
|
|
|
|
SkASSERT(options.srgb);
|
|
|
|
colorType = kRGBA_F16_SkColorType;
|
2017-02-07 18:56:11 +00:00
|
|
|
colorSpace = SkColorSpace::MakeSRGBLinear();
|
2016-11-29 22:14:58 +00:00
|
|
|
} else if (options.srgb) {
|
2017-02-07 18:56:11 +00:00
|
|
|
colorSpace = SkColorSpace::MakeSRGB();
|
2016-11-29 22:14:58 +00:00
|
|
|
}
|
|
|
|
SkImageInfo info = SkImageInfo::Make(options.size.width(), options.size.height(), colorType,
|
|
|
|
kPremul_SkAlphaType, colorSpace);
|
2015-12-10 15:52:45 +00:00
|
|
|
if (options.raster) {
|
2016-11-29 22:14:58 +00:00
|
|
|
auto rasterSurface = SkSurface::MakeRaster(info);
|
2016-05-23 16:02:38 +00:00
|
|
|
srand(0);
|
2017-06-14 19:54:32 +00:00
|
|
|
draw(prepare_canvas(rasterSurface->getCanvas()));
|
2017-07-11 20:03:13 +00:00
|
|
|
rasterData = encode_snapshot(rasterSurface);
|
2015-12-10 15:52:45 +00:00
|
|
|
}
|
2020-03-19 19:54:28 +00:00
|
|
|
#ifdef SK_GL
|
2015-12-10 15:52:45 +00:00
|
|
|
if (options.gpu) {
|
2018-05-29 17:56:27 +00:00
|
|
|
std::unique_ptr<sk_gpu_test::GLTestContext> glContext;
|
2020-07-13 20:13:31 +00:00
|
|
|
sk_sp<GrDirectContext> direct = create_direct_context(gGLDriverInfo, &glContext);
|
|
|
|
if (!direct) {
|
2016-07-27 18:17:18 +00:00
|
|
|
fputs("Unable to get GrContext.\n", stderr);
|
2015-12-10 15:52:45 +00:00
|
|
|
} else {
|
2020-07-13 20:13:31 +00:00
|
|
|
if (!setup_backend_objects(direct.get(), source, options)) {
|
2017-11-16 19:59:48 +00:00
|
|
|
fputs("Unable to create backend objects.\n", stderr);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2020-07-13 20:13:31 +00:00
|
|
|
auto surface = SkSurface::MakeRenderTarget(direct.get(), SkBudgeted::kNo, info);
|
2015-12-10 15:52:45 +00:00
|
|
|
if (!surface) {
|
|
|
|
fputs("Unable to get render surface.\n", stderr);
|
|
|
|
exit(1);
|
|
|
|
}
|
2016-05-23 16:02:38 +00:00
|
|
|
srand(0);
|
2017-06-14 19:54:32 +00:00
|
|
|
draw(prepare_canvas(surface->getCanvas()));
|
2017-07-11 20:03:13 +00:00
|
|
|
gpuData = encode_snapshot(surface);
|
2015-12-10 15:52:45 +00:00
|
|
|
}
|
|
|
|
}
|
2020-03-19 19:54:28 +00:00
|
|
|
#endif
|
2020-12-08 16:51:02 +00:00
|
|
|
|
|
|
|
#ifdef SK_SUPPORT_PDF
|
2015-12-10 15:52:45 +00:00
|
|
|
if (options.pdf) {
|
|
|
|
SkDynamicMemoryWStream pdfStream;
|
2019-01-07 15:00:48 +00:00
|
|
|
auto document = SkPDF::MakeDocument(&pdfStream);
|
2016-08-31 11:58:19 +00:00
|
|
|
if (document) {
|
|
|
|
srand(0);
|
2017-06-14 19:54:32 +00:00
|
|
|
draw(prepare_canvas(document->beginPage(options.size.width(), options.size.height())));
|
2016-08-31 11:58:19 +00:00
|
|
|
document->close();
|
2016-09-12 19:01:44 +00:00
|
|
|
pdfData = pdfStream.detachAsData();
|
2016-08-31 11:58:19 +00:00
|
|
|
}
|
2015-12-10 15:52:45 +00:00
|
|
|
}
|
2020-12-08 16:51:02 +00:00
|
|
|
#endif
|
|
|
|
|
2015-12-10 15:52:45 +00:00
|
|
|
if (options.skp) {
|
2019-12-04 15:33:52 +00:00
|
|
|
auto size = SkSize::Make(options.size);
|
2015-12-10 15:52:45 +00:00
|
|
|
SkPictureRecorder recorder;
|
2016-05-23 16:02:38 +00:00
|
|
|
srand(0);
|
2017-06-14 19:54:32 +00:00
|
|
|
draw(prepare_canvas(recorder.beginRecording(size.width(), size.height())));
|
2016-03-25 18:29:34 +00:00
|
|
|
auto picture = recorder.finishRecordingAsPicture();
|
2015-12-10 15:52:45 +00:00
|
|
|
SkDynamicMemoryWStream skpStream;
|
|
|
|
picture->serialize(&skpStream);
|
2016-09-12 19:01:44 +00:00
|
|
|
skpData = skpStream.detachAsData();
|
2015-12-10 15:52:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
printf("{\n");
|
2017-02-13 18:55:17 +00:00
|
|
|
if (!options.textOnly) {
|
2017-06-01 17:24:11 +00:00
|
|
|
dump_output(rasterData, "Raster", false);
|
|
|
|
dump_output(gpuData, "Gpu", false);
|
|
|
|
dump_output(pdfData, "Pdf", false);
|
|
|
|
dump_output(skpData, "Skp", false);
|
2017-02-13 18:55:17 +00:00
|
|
|
} else {
|
2017-02-15 15:20:30 +00:00
|
|
|
std::string textoutput = gTextOutput.str();
|
2017-06-01 17:24:11 +00:00
|
|
|
dump_output(textoutput.c_str(), textoutput.length(), "Text", false);
|
2017-02-13 18:55:17 +00:00
|
|
|
}
|
2017-06-01 17:24:11 +00:00
|
|
|
std::string glinfo = gGLDriverInfo.str();
|
|
|
|
dump_output(glinfo.c_str(), glinfo.length(), "GLInfo", true);
|
2015-12-10 15:52:45 +00:00
|
|
|
printf("}\n");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|