dd9003afda
Before, we could only wait for all reserved objects to serialize. The new unit test `SkPDF_abort_jobs` crashed before this change. Change-Id: Ia2fcb33251c6c32c125f631ed3eeb619f616bef3 Reviewed-on: https://skia-review.googlesource.com/c/179856 Auto-Submit: Hal Canary <halcanary@google.com> Commit-Queue: Herb Derby <herb@google.com> Reviewed-by: Herb Derby <herb@google.com>
259 lines
8.0 KiB
C++
259 lines
8.0 KiB
C++
/*
|
||
* Copyright 2013 Google Inc.
|
||
*
|
||
* Use of this source code is governed by a BSD-style license that can be
|
||
* found in the LICENSE file.
|
||
*/
|
||
#include "Test.h"
|
||
|
||
#include "Resources.h"
|
||
#include "SkCanvas.h"
|
||
#include "SkExecutor.h"
|
||
#include "SkOSFile.h"
|
||
#include "SkOSPath.h"
|
||
#include "SkPDFDocument.h"
|
||
#include "SkStream.h"
|
||
|
||
#include "sk_tool_utils.h"
|
||
|
||
static void test_empty(skiatest::Reporter* reporter) {
|
||
SkDynamicMemoryWStream stream;
|
||
|
||
sk_sp<SkDocument> doc(SkPDF::MakeDocument(&stream));
|
||
|
||
doc->close();
|
||
|
||
REPORTER_ASSERT(reporter, stream.bytesWritten() == 0);
|
||
}
|
||
|
||
static void test_abort(skiatest::Reporter* reporter) {
|
||
SkDynamicMemoryWStream stream;
|
||
sk_sp<SkDocument> doc(SkPDF::MakeDocument(&stream));
|
||
|
||
SkCanvas* canvas = doc->beginPage(100, 100);
|
||
canvas->drawColor(SK_ColorRED);
|
||
doc->endPage();
|
||
|
||
doc->abort();
|
||
|
||
// Test that only the header is written, not the full document.
|
||
REPORTER_ASSERT(reporter, stream.bytesWritten() < 256);
|
||
}
|
||
|
||
static void test_abortWithFile(skiatest::Reporter* reporter) {
|
||
SkString tmpDir = skiatest::GetTmpDir();
|
||
|
||
if (tmpDir.isEmpty()) {
|
||
ERRORF(reporter, "missing tmpDir.");
|
||
return;
|
||
}
|
||
|
||
SkString path = SkOSPath::Join(tmpDir.c_str(), "aborted.pdf");
|
||
if (!SkFILEWStream(path.c_str()).isValid()) {
|
||
ERRORF(reporter, "unable to write to: %s", path.c_str());
|
||
return;
|
||
}
|
||
|
||
// Make sure doc's destructor is called to flush.
|
||
{
|
||
SkFILEWStream stream(path.c_str());
|
||
sk_sp<SkDocument> doc = SkPDF::MakeDocument(&stream);
|
||
|
||
SkCanvas* canvas = doc->beginPage(100, 100);
|
||
canvas->drawColor(SK_ColorRED);
|
||
doc->endPage();
|
||
|
||
doc->abort();
|
||
}
|
||
|
||
FILE* file = fopen(path.c_str(), "r");
|
||
// Test that only the header is written, not the full document.
|
||
char buffer[256];
|
||
REPORTER_ASSERT(reporter, fread(buffer, 1, sizeof(buffer), file) < sizeof(buffer));
|
||
fclose(file);
|
||
}
|
||
|
||
static void test_file(skiatest::Reporter* reporter) {
|
||
SkString tmpDir = skiatest::GetTmpDir();
|
||
if (tmpDir.isEmpty()) {
|
||
ERRORF(reporter, "missing tmpDir.");
|
||
return;
|
||
}
|
||
|
||
SkString path = SkOSPath::Join(tmpDir.c_str(), "file.pdf");
|
||
if (!SkFILEWStream(path.c_str()).isValid()) {
|
||
ERRORF(reporter, "unable to write to: %s", path.c_str());
|
||
return;
|
||
}
|
||
|
||
{
|
||
SkFILEWStream stream(path.c_str());
|
||
sk_sp<SkDocument> doc = SkPDF::MakeDocument(&stream);
|
||
SkCanvas* canvas = doc->beginPage(100, 100);
|
||
|
||
canvas->drawColor(SK_ColorRED);
|
||
doc->endPage();
|
||
doc->close();
|
||
}
|
||
|
||
FILE* file = fopen(path.c_str(), "r");
|
||
REPORTER_ASSERT(reporter, file != nullptr);
|
||
char header[100];
|
||
REPORTER_ASSERT(reporter, fread(header, 4, 1, file) != 0);
|
||
REPORTER_ASSERT(reporter, strncmp(header, "%PDF", 4) == 0);
|
||
fclose(file);
|
||
}
|
||
|
||
static void test_close(skiatest::Reporter* reporter) {
|
||
SkDynamicMemoryWStream stream;
|
||
sk_sp<SkDocument> doc(SkPDF::MakeDocument(&stream));
|
||
|
||
SkCanvas* canvas = doc->beginPage(100, 100);
|
||
canvas->drawColor(SK_ColorRED);
|
||
doc->endPage();
|
||
|
||
doc->close();
|
||
|
||
REPORTER_ASSERT(reporter, stream.bytesWritten() != 0);
|
||
}
|
||
|
||
DEF_TEST(SkPDF_document_tests, reporter) {
|
||
REQUIRE_PDF_DOCUMENT(document_tests, reporter);
|
||
test_empty(reporter);
|
||
test_abort(reporter);
|
||
test_abortWithFile(reporter);
|
||
test_file(reporter);
|
||
test_close(reporter);
|
||
}
|
||
|
||
DEF_TEST(SkPDF_document_skbug_4734, r) {
|
||
REQUIRE_PDF_DOCUMENT(SkPDF_document_skbug_4734, r);
|
||
SkDynamicMemoryWStream stream;
|
||
sk_sp<SkDocument> doc(SkPDF::MakeDocument(&stream));
|
||
SkCanvas* canvas = doc->beginPage(64, 64);
|
||
canvas->scale(10000.0f, 10000.0f);
|
||
canvas->translate(20.0f, 10.0f);
|
||
canvas->rotate(30.0f);
|
||
const char text[] = "HELLO";
|
||
canvas->drawString(text, 0, 0, SkPaint());
|
||
}
|
||
|
||
static bool contains(const uint8_t* result, size_t size, const char expectation[]) {
|
||
size_t len = strlen(expectation);
|
||
size_t N = 1 + size - len;
|
||
for (size_t i = 0; i < N; ++i) {
|
||
if (0 == memcmp(result + i, expectation, len)) {
|
||
return true;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
// verify that the PDFA flag does something.
|
||
DEF_TEST(SkPDF_pdfa_document, r) {
|
||
REQUIRE_PDF_DOCUMENT(SkPDF_pdfa_document, r);
|
||
|
||
SkPDF::Metadata pdfMetadata;
|
||
pdfMetadata.fTitle = "test document";
|
||
pdfMetadata.fCreation = {0, 1999, 12, 5, 31, 23, 59, 59};
|
||
pdfMetadata.fPDFA = true;
|
||
|
||
SkDynamicMemoryWStream buffer;
|
||
auto doc = SkPDF::MakeDocument(&buffer, pdfMetadata);
|
||
doc->beginPage(64, 64)->drawColor(SK_ColorRED);
|
||
doc->close();
|
||
sk_sp<SkData> data(buffer.detachAsData());
|
||
|
||
static const char* expectations[] = {
|
||
"sRGB IEC61966-2.1",
|
||
"<dc:title><rdf:Alt><rdf:li xml:lang=\"x-default\">test document",
|
||
"<xmp:CreateDate>1999-12-31T23:59:59+00:00</xmp:CreateDate>",
|
||
"/Subtype /XML",
|
||
"/CreationDate (D:19991231235959+00'00')>>",
|
||
};
|
||
for (const char* expectation : expectations) {
|
||
if (!contains(data->bytes(), data->size(), expectation)) {
|
||
ERRORF(r, "PDFA expectation missing: '%s'.", expectation);
|
||
}
|
||
}
|
||
pdfMetadata.fProducer = "phoney library";
|
||
pdfMetadata.fPDFA = true;
|
||
doc = SkPDF::MakeDocument(&buffer, pdfMetadata);
|
||
doc->beginPage(64, 64)->drawColor(SK_ColorRED);
|
||
doc->close();
|
||
data = buffer.detachAsData();
|
||
|
||
static const char* moreExpectations[] = {
|
||
"/Producer (phoney library)",
|
||
"/ProductionLibrary (Skia/PDF m",
|
||
"<!-- <skia:ProductionLibrary>Skia/PDF m",
|
||
"<pdf:Producer>phoney library</pdf:Producer>",
|
||
};
|
||
for (const char* expectation : moreExpectations) {
|
||
if (!contains(data->bytes(), data->size(), expectation)) {
|
||
ERRORF(r, "PDFA expectation missing: '%s'.", expectation);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
DEF_TEST(SkPDF_unicode_metadata, r) {
|
||
REQUIRE_PDF_DOCUMENT(SkPDF_unicode_metadata, r);
|
||
SkPDF::Metadata pdfMetadata;
|
||
pdfMetadata.fTitle = "𝓐𝓑𝓒𝓓𝓔 𝓕𝓖𝓗𝓘𝓙"; // Out of basic multilingual plane
|
||
pdfMetadata.fAuthor = "ABCDE FGHIJ"; // ASCII
|
||
pdfMetadata.fSubject = "αβγδε ζηθικ"; // inside basic multilingual plane
|
||
pdfMetadata.fPDFA = true;
|
||
SkDynamicMemoryWStream wStream;
|
||
{
|
||
auto doc = SkPDF::MakeDocument(&wStream, pdfMetadata);
|
||
doc->beginPage(612, 792)->drawColor(SK_ColorCYAN);
|
||
}
|
||
sk_sp<SkData> data(wStream.detachAsData());
|
||
static const char* expectations[] = {
|
||
"<</Title <FEFFD835DCD0D835DCD1D835DCD2D835DCD3D835DCD40020"
|
||
"D835DCD5D835DCD6D835DCD7D835DCD8D835DCD9>",
|
||
"/Author (ABCDE FGHIJ)",
|
||
"Subject <FEFF03B103B203B303B403B5002003B603B703B803B903BA>",
|
||
};
|
||
for (const char* expectation : expectations) {
|
||
if (!contains(data->bytes(), data->size(), expectation)) {
|
||
ERRORF(r, "PDF expectation missing: '%s'.", expectation);
|
||
}
|
||
}
|
||
}
|
||
|
||
// Make sure we excercise the multi-page functionality without problems.
|
||
// Add this to args.gn to output the PDF to a file:
|
||
// extra_cflags = [ "-DSK_PDF_TEST_MULTIPAGE=\"/tmp/skpdf_test_multipage.pdf\"" ]
|
||
DEF_TEST(SkPDF_multiple_pages, r) {
|
||
REQUIRE_PDF_DOCUMENT(SkPDF_multiple_pages, r);
|
||
int n = 100;
|
||
#ifdef SK_PDF_TEST_MULTIPAGE
|
||
SkFILEWStream wStream(SK_PDF_TEST_MULTIPAGE);
|
||
#else
|
||
SkDynamicMemoryWStream wStream;
|
||
#endif
|
||
auto doc = SkPDF::MakeDocument(&wStream);
|
||
for (int i = 0; i < n; ++i) {
|
||
doc->beginPage(612, 792)->drawColor(
|
||
SkColorSetARGB(0xFF, 0x00, (uint8_t)(255.0f * i / (n - 1)), 0x00));
|
||
}
|
||
}
|
||
|
||
// Test to make sure that jobs launched by PDF backend don't cause a segfault
|
||
// after calling abort().
|
||
DEF_TEST(SkPDF_abort_jobs, rep) {
|
||
SkBitmap b;
|
||
b.allocN32Pixels(612, 792);
|
||
b.eraseColor(0x4F9643A0);
|
||
SkPDF::Metadata metadata;
|
||
std::unique_ptr<SkExecutor> executor = SkExecutor::MakeFIFOThreadPool();
|
||
metadata.fExecutor = executor.get();
|
||
SkNullWStream dst;
|
||
sk_sp<SkDocument> doc = SkPDF::MakeDocument(&dst, metadata);
|
||
doc->beginPage(612, 792)->drawBitmap(b, 0, 0);
|
||
doc->abort();
|
||
}
|
||
|