| |
| /* |
| * Copyright 2011 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| |
| #include "SkPDFCatalog.h" |
| #include "SkPDFTypes.h" |
| #include "SkStream.h" |
| |
| #ifdef SK_BUILD_FOR_WIN |
| #define SNPRINTF _snprintf |
| #else |
| #define SNPRINTF snprintf |
| #endif |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| void SkPDFObject::emit(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| SkPDFObject* realObject = catalog->getSubstituteObject(this); |
| return realObject->emitObject(stream, catalog, indirect); |
| } |
| |
| size_t SkPDFObject::getOutputSize(SkPDFCatalog* catalog, bool indirect) { |
| SkDynamicMemoryWStream buffer; |
| emit(&buffer, catalog, indirect); |
| return buffer.getOffset(); |
| } |
| |
| void SkPDFObject::getResources(const SkTSet<SkPDFObject*>& knownResourceObjects, |
| SkTSet<SkPDFObject*>* newResourceObjects) {} |
| |
| void SkPDFObject::emitIndirectObject(SkWStream* stream, SkPDFCatalog* catalog) { |
| catalog->emitObjectNumber(stream, this); |
| stream->writeText(" obj\n"); |
| emit(stream, catalog, false); |
| stream->writeText("\nendobj\n"); |
| } |
| |
| size_t SkPDFObject::getIndirectOutputSize(SkPDFCatalog* catalog) { |
| return catalog->getObjectNumberSize(this) + strlen(" obj\n") + |
| this->getOutputSize(catalog, false) + strlen("\nendobj\n"); |
| } |
| |
| void SkPDFObject::AddResourceHelper(SkPDFObject* resource, |
| SkTDArray<SkPDFObject*>* list) { |
| list->push(resource); |
| resource->ref(); |
| } |
| |
| void SkPDFObject::GetResourcesHelper( |
| const SkTDArray<SkPDFObject*>* resources, |
| const SkTSet<SkPDFObject*>& knownResourceObjects, |
| SkTSet<SkPDFObject*>* newResourceObjects) { |
| if (resources->count()) { |
| newResourceObjects->setReserve( |
| newResourceObjects->count() + resources->count()); |
| for (int i = 0; i < resources->count(); i++) { |
| if (!knownResourceObjects.contains((*resources)[i]) && |
| !newResourceObjects->contains((*resources)[i])) { |
| newResourceObjects->add((*resources)[i]); |
| (*resources)[i]->ref(); |
| (*resources)[i]->getResources(knownResourceObjects, |
| newResourceObjects); |
| } |
| } |
| } |
| } |
| |
| SkPDFObjRef::SkPDFObjRef(SkPDFObject* obj) : fObj(obj) { |
| SkSafeRef(obj); |
| } |
| |
| SkPDFObjRef::~SkPDFObjRef() {} |
| |
| void SkPDFObjRef::emitObject(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| SkASSERT(!indirect); |
| catalog->emitObjectNumber(stream, fObj.get()); |
| stream->writeText(" R"); |
| } |
| |
| size_t SkPDFObjRef::getOutputSize(SkPDFCatalog* catalog, bool indirect) { |
| SkASSERT(!indirect); |
| return catalog->getObjectNumberSize(fObj.get()) + strlen(" R"); |
| } |
| |
| SkPDFInt::SkPDFInt(int32_t value) : fValue(value) {} |
| SkPDFInt::~SkPDFInt() {} |
| |
| void SkPDFInt::emitObject(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| if (indirect) { |
| return emitIndirectObject(stream, catalog); |
| } |
| stream->writeDecAsText(fValue); |
| } |
| |
| SkPDFBool::SkPDFBool(bool value) : fValue(value) {} |
| SkPDFBool::~SkPDFBool() {} |
| |
| void SkPDFBool::emitObject(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| SkASSERT(!indirect); |
| if (fValue) { |
| stream->writeText("true"); |
| } else { |
| stream->writeText("false"); |
| } |
| } |
| |
| size_t SkPDFBool::getOutputSize(SkPDFCatalog* catalog, bool indirect) { |
| SkASSERT(!indirect); |
| if (fValue) { |
| return strlen("true"); |
| } |
| return strlen("false"); |
| } |
| |
| SkPDFScalar::SkPDFScalar(SkScalar value) : fValue(value) {} |
| SkPDFScalar::~SkPDFScalar() {} |
| |
| void SkPDFScalar::emitObject(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| if (indirect) { |
| return emitIndirectObject(stream, catalog); |
| } |
| |
| Append(fValue, stream); |
| } |
| |
| // static |
| void SkPDFScalar::Append(SkScalar value, SkWStream* stream) { |
| // The range of reals in PDF/A is the same as SkFixed: +/- 32,767 and |
| // +/- 1/65,536 (though integers can range from 2^31 - 1 to -2^31). |
| // When using floats that are outside the whole value range, we can use |
| // integers instead. |
| |
| #if !defined(SK_ALLOW_LARGE_PDF_SCALARS) |
| if (value > 32767 || value < -32767) { |
| stream->writeDecAsText(SkScalarRoundToInt(value)); |
| return; |
| } |
| |
| char buffer[SkStrAppendScalar_MaxSize]; |
| char* end = SkStrAppendFixed(buffer, SkScalarToFixed(value)); |
| stream->write(buffer, end - buffer); |
| return; |
| #endif // !SK_ALLOW_LARGE_PDF_SCALARS |
| |
| #if defined(SK_ALLOW_LARGE_PDF_SCALARS) |
| // Floats have 24bits of significance, so anything outside that range is |
| // no more precise than an int. (Plus PDF doesn't support scientific |
| // notation, so this clamps to SK_Max/MinS32). |
| if (value > (1 << 24) || value < -(1 << 24)) { |
| stream->writeDecAsText(value); |
| return; |
| } |
| // Continue to enforce the PDF limits for small floats. |
| if (value < 1.0f/65536 && value > -1.0f/65536) { |
| stream->writeDecAsText(0); |
| return; |
| } |
| // SkStrAppendFloat might still use scientific notation, so use snprintf |
| // directly.. |
| static const int kFloat_MaxSize = 19; |
| char buffer[kFloat_MaxSize]; |
| int len = SNPRINTF(buffer, kFloat_MaxSize, "%#.8f", value); |
| // %f always prints trailing 0s, so strip them. |
| for (; buffer[len - 1] == '0' && len > 0; len--) { |
| buffer[len - 1] = '\0'; |
| } |
| if (buffer[len - 1] == '.') { |
| buffer[len - 1] = '\0'; |
| } |
| stream->writeText(buffer); |
| return; |
| #endif // SK_ALLOW_LARGE_PDF_SCALARS |
| } |
| |
| SkPDFString::SkPDFString(const char value[]) |
| : fValue(FormatString(value, strlen(value))) { |
| } |
| |
| SkPDFString::SkPDFString(const SkString& value) |
| : fValue(FormatString(value.c_str(), value.size())) { |
| } |
| |
| SkPDFString::SkPDFString(const uint16_t* value, size_t len, bool wideChars) |
| : fValue(FormatString(value, len, wideChars)) { |
| } |
| |
| SkPDFString::~SkPDFString() {} |
| |
| void SkPDFString::emitObject(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| if (indirect) |
| return emitIndirectObject(stream, catalog); |
| stream->write(fValue.c_str(), fValue.size()); |
| } |
| |
| size_t SkPDFString::getOutputSize(SkPDFCatalog* catalog, bool indirect) { |
| if (indirect) |
| return getIndirectOutputSize(catalog); |
| return fValue.size(); |
| } |
| |
| // static |
| SkString SkPDFString::FormatString(const char* input, size_t len) { |
| return DoFormatString(input, len, false, false); |
| } |
| |
| SkString SkPDFString::FormatString(const uint16_t* input, size_t len, |
| bool wideChars) { |
| return DoFormatString(input, len, true, wideChars); |
| } |
| |
| // static |
| SkString SkPDFString::DoFormatString(const void* input, size_t len, |
| bool wideInput, bool wideOutput) { |
| SkASSERT(len <= kMaxLen); |
| const uint16_t* win = (const uint16_t*) input; |
| const char* cin = (const char*) input; |
| |
| if (wideOutput) { |
| SkASSERT(wideInput); |
| SkString result; |
| result.append("<"); |
| for (size_t i = 0; i < len; i++) { |
| result.appendHex(win[i], 4); |
| } |
| result.append(">"); |
| return result; |
| } |
| |
| // 7-bit clean is a heuristic to decide what string format to use; |
| // a 7-bit clean string should require little escaping. |
| bool sevenBitClean = true; |
| for (size_t i = 0; i < len; i++) { |
| SkASSERT(!wideInput || !(win[i] & ~0xFF)); |
| char val = wideInput ? win[i] : cin[i]; |
| if (val > '~' || val < ' ') { |
| sevenBitClean = false; |
| break; |
| } |
| } |
| |
| SkString result; |
| if (sevenBitClean) { |
| result.append("("); |
| for (size_t i = 0; i < len; i++) { |
| SkASSERT(!wideInput || !(win[i] & ~0xFF)); |
| char val = wideInput ? win[i] : cin[i]; |
| if (val == '\\' || val == '(' || val == ')') { |
| result.append("\\"); |
| } |
| result.append(&val, 1); |
| } |
| result.append(")"); |
| } else { |
| result.append("<"); |
| for (size_t i = 0; i < len; i++) { |
| SkASSERT(!wideInput || !(win[i] & ~0xFF)); |
| unsigned char val = wideInput ? win[i] : cin[i]; |
| result.appendHex(val, 2); |
| } |
| result.append(">"); |
| } |
| |
| return result; |
| } |
| |
| SkPDFName::SkPDFName(const char name[]) : fValue(FormatName(SkString(name))) {} |
| SkPDFName::SkPDFName(const SkString& name) : fValue(FormatName(name)) {} |
| SkPDFName::~SkPDFName() {} |
| |
| bool SkPDFName::operator==(const SkPDFName& b) const { |
| return fValue == b.fValue; |
| } |
| |
| void SkPDFName::emitObject(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| SkASSERT(!indirect); |
| stream->write(fValue.c_str(), fValue.size()); |
| } |
| |
| size_t SkPDFName::getOutputSize(SkPDFCatalog* catalog, bool indirect) { |
| SkASSERT(!indirect); |
| return fValue.size(); |
| } |
| |
| // static |
| SkString SkPDFName::FormatName(const SkString& input) { |
| SkASSERT(input.size() <= kMaxLen); |
| // TODO(vandebo) If more escaping is needed, improve the linear scan. |
| static const char escaped[] = "#/%()<>[]{}"; |
| |
| SkString result("/"); |
| for (size_t i = 0; i < input.size(); i++) { |
| if (input[i] & 0x80 || input[i] < '!' || strchr(escaped, input[i])) { |
| result.append("#"); |
| // Mask with 0xFF to avoid sign extension. i.e. #FFFFFF81 |
| result.appendHex(input[i] & 0xFF, 2); |
| } else { |
| result.append(input.c_str() + i, 1); |
| } |
| } |
| |
| return result; |
| } |
| |
| SkPDFArray::SkPDFArray() {} |
| SkPDFArray::~SkPDFArray() { |
| fValue.unrefAll(); |
| } |
| |
| void SkPDFArray::emitObject(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| if (indirect) { |
| return emitIndirectObject(stream, catalog); |
| } |
| |
| stream->writeText("["); |
| for (int i = 0; i < fValue.count(); i++) { |
| fValue[i]->emit(stream, catalog, false); |
| if (i + 1 < fValue.count()) { |
| stream->writeText(" "); |
| } |
| } |
| stream->writeText("]"); |
| } |
| |
| size_t SkPDFArray::getOutputSize(SkPDFCatalog* catalog, bool indirect) { |
| if (indirect) { |
| return getIndirectOutputSize(catalog); |
| } |
| |
| size_t result = strlen("[]"); |
| if (fValue.count()) { |
| result += fValue.count() - 1; |
| } |
| for (int i = 0; i < fValue.count(); i++) { |
| result += fValue[i]->getOutputSize(catalog, false); |
| } |
| return result; |
| } |
| |
| void SkPDFArray::reserve(int length) { |
| SkASSERT(length <= kMaxLen); |
| fValue.setReserve(length); |
| } |
| |
| SkPDFObject* SkPDFArray::setAt(int offset, SkPDFObject* value) { |
| SkASSERT(offset < fValue.count()); |
| value->ref(); |
| fValue[offset]->unref(); |
| fValue[offset] = value; |
| return value; |
| } |
| |
| SkPDFObject* SkPDFArray::append(SkPDFObject* value) { |
| SkASSERT(fValue.count() < kMaxLen); |
| value->ref(); |
| fValue.push(value); |
| return value; |
| } |
| |
| void SkPDFArray::appendInt(int32_t value) { |
| SkASSERT(fValue.count() < kMaxLen); |
| fValue.push(new SkPDFInt(value)); |
| } |
| |
| void SkPDFArray::appendScalar(SkScalar value) { |
| SkASSERT(fValue.count() < kMaxLen); |
| fValue.push(new SkPDFScalar(value)); |
| } |
| |
| void SkPDFArray::appendName(const char name[]) { |
| SkASSERT(fValue.count() < kMaxLen); |
| fValue.push(new SkPDFName(name)); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| SkPDFDict::SkPDFDict() {} |
| |
| SkPDFDict::SkPDFDict(const char type[]) { |
| insertName("Type", type); |
| } |
| |
| SkPDFDict::~SkPDFDict() { |
| clear(); |
| } |
| |
| int SkPDFDict::size() const { |
| SkAutoMutexAcquire lock(fMutex); |
| return fValue.count(); |
| } |
| |
| |
| void SkPDFDict::emitObject(SkWStream* stream, SkPDFCatalog* catalog, |
| bool indirect) { |
| if (indirect) { |
| return emitIndirectObject(stream, catalog); |
| } |
| |
| SkAutoMutexAcquire lock(fMutex); // If another thread triggers a |
| // resize while this thread is in |
| // the for-loop, we can be left |
| // with a bad fValue[i] reference. |
| stream->writeText("<<"); |
| for (int i = 0; i < fValue.count(); i++) { |
| SkASSERT(fValue[i].key); |
| SkASSERT(fValue[i].value); |
| fValue[i].key->emitObject(stream, catalog, false); |
| stream->writeText(" "); |
| fValue[i].value->emit(stream, catalog, false); |
| stream->writeText("\n"); |
| } |
| stream->writeText(">>"); |
| } |
| |
| size_t SkPDFDict::getOutputSize(SkPDFCatalog* catalog, bool indirect) { |
| if (indirect) { |
| return getIndirectOutputSize(catalog); |
| } |
| |
| SkAutoMutexAcquire lock(fMutex); // If another thread triggers a |
| // resize while this thread is in |
| // the for-loop, we can be left |
| // with a bad fValue[i] reference. |
| size_t result = strlen("<<>>") + (fValue.count() * 2); |
| for (int i = 0; i < fValue.count(); i++) { |
| SkASSERT(fValue[i].key); |
| SkASSERT(fValue[i].value); |
| result += fValue[i].key->getOutputSize(catalog, false); |
| result += fValue[i].value->getOutputSize(catalog, false); |
| } |
| return result; |
| } |
| |
| SkPDFObject* SkPDFDict::append(SkPDFName* key, SkPDFObject* value) { |
| SkASSERT(key); |
| SkASSERT(value); |
| SkAutoMutexAcquire lock(fMutex); // If the SkTDArray resizes while |
| // two threads access array, one |
| // is left with a bad pointer. |
| *(fValue.append()) = Rec(key, value); |
| return value; |
| } |
| |
| SkPDFObject* SkPDFDict::insert(SkPDFName* key, SkPDFObject* value) { |
| return this->append(SkRef(key), SkRef(value)); |
| } |
| |
| SkPDFObject* SkPDFDict::insert(const char key[], SkPDFObject* value) { |
| return this->append(new SkPDFName(key), SkRef(value)); |
| } |
| |
| void SkPDFDict::insertInt(const char key[], int32_t value) { |
| (void)this->append(new SkPDFName(key), new SkPDFInt(value)); |
| } |
| |
| void SkPDFDict::insertScalar(const char key[], SkScalar value) { |
| (void)this->append(new SkPDFName(key), new SkPDFScalar(value)); |
| } |
| |
| void SkPDFDict::insertName(const char key[], const char name[]) { |
| (void)this->append(new SkPDFName(key), new SkPDFName(name)); |
| } |
| |
| void SkPDFDict::clear() { |
| SkAutoMutexAcquire lock(fMutex); |
| for (int i = 0; i < fValue.count(); i++) { |
| SkASSERT(fValue[i].key); |
| SkASSERT(fValue[i].value); |
| fValue[i].key->unref(); |
| fValue[i].value->unref(); |
| } |
| fValue.reset(); |
| } |
| |
| void SkPDFDict::remove(const char key[]) { |
| SkASSERT(key); |
| SkPDFName name(key); |
| SkAutoMutexAcquire lock(fMutex); |
| for (int i = 0; i < fValue.count(); i++) { |
| SkASSERT(fValue[i].key); |
| if (*(fValue[i].key) == name) { |
| fValue[i].key->unref(); |
| SkASSERT(fValue[i].value); |
| fValue[i].value->unref(); |
| fValue.removeShuffle(i); |
| return; |
| } |
| } |
| } |
| |
| void SkPDFDict::mergeFrom(const SkPDFDict& other) { |
| SkAutoMutexAcquire lockOther(other.fMutex); |
| SkTDArray<Rec> copy(other.fValue); |
| lockOther.release(); // Do not hold both mutexes at once. |
| |
| SkAutoMutexAcquire lock(fMutex); |
| for (int i = 0; i < copy.count(); i++) { |
| *(fValue.append()) = Rec(SkRef(copy[i].key), SkRef(copy[i].value)); |
| } |
| } |