From d52bf699d08efb381fb5e329af2d57e127f7bcd6 Mon Sep 17 00:00:00 2001 From: Adrian Johnson Date: Sun, 12 Oct 2014 15:08:39 +1030 Subject: [PATCH] PS: emit icc color spaces --- poppler/GfxState.cc | 103 ++++++++++++++++++++++++++++++++----------------- poppler/GfxState.h | 9 ++++- poppler/PSOutputDev.cc | 67 ++++++++++++++++++++++++-------- poppler/PSOutputDev.h | 8 ++-- 4 files changed, 130 insertions(+), 57 deletions(-) diff --git a/poppler/GfxState.cc b/poppler/GfxState.cc index eae85dc..10e2d63 100644 --- a/poppler/GfxState.cc +++ b/poppler/GfxState.cc @@ -203,7 +203,9 @@ void GfxColorTransform::doTransform(void *in, void *out, unsigned int size) { } // transformA should be a cmsHTRANSFORM -GfxColorTransform::GfxColorTransform(void *transformA, int cmsIntentA, unsigned int transformPixelTypeA) { +GfxColorTransform::GfxColorTransform(void *sourceProfileA, void *transformA, + int cmsIntentA, unsigned int transformPixelTypeA) { + sourceProfile = sourceProfileA; transform = transformA; refCount = 1; cmsIntent = cmsIntentA; @@ -211,6 +213,8 @@ GfxColorTransform::GfxColorTransform(void *transformA, int cmsIntentA, unsigned } GfxColorTransform::~GfxColorTransform() { + if (sourceProfile) + cmsCloseProfile(sourceProfile); cmsDeleteTransform(transform); } @@ -250,9 +254,8 @@ void GfxColorSpace::setDisplayProfile(void *displayProfileA) { INTENT_RELATIVE_COLORIMETRIC,LCMS_FLAGS)) == 0) { error(errSyntaxWarning, -1, "Can't create Lab transform"); } else { - XYZ2DisplayTransform = new GfxColorTransform(transform, INTENT_RELATIVE_COLORIMETRIC, displayPixelType); + XYZ2DisplayTransform = new GfxColorTransform(displayProfile, transform, INTENT_RELATIVE_COLORIMETRIC, displayPixelType); } - cmsCloseProfile(XYZProfile); } } @@ -451,8 +454,9 @@ int GfxColorSpace::setupColorProfiles() CHANNELS_SH(nChannels) | BYTES_SH(1), INTENT_RELATIVE_COLORIMETRIC,LCMS_FLAGS)) == 0) { error(errSyntaxWarning, -1, "Can't create Lab transform"); + cmsCloseProfile(XYZProfile); } else { - XYZ2DisplayTransform = new GfxColorTransform(transform, INTENT_RELATIVE_COLORIMETRIC, displayPixelType); + XYZ2DisplayTransform = new GfxColorTransform(XYZProfile, transform, INTENT_RELATIVE_COLORIMETRIC, displayPixelType); } cmsCloseProfile(XYZProfile); } @@ -1830,18 +1834,8 @@ GfxColorSpace *GfxICCBasedColorSpace::parse(Array *arr, OutputDev *out, GfxState cs = static_cast(item->cs->copy()); int transformIntent = cs->getIntent(); int cmsIntent = INTENT_RELATIVE_COLORIMETRIC; - if (state != NULL) { - const char *intent = state->getRenderingIntent(); - if (intent != NULL) { - if (strcmp(intent, "AbsoluteColorimetric") == 0) { - cmsIntent = INTENT_ABSOLUTE_COLORIMETRIC; - } else if (strcmp(intent, "Saturation") == 0) { - cmsIntent = INTENT_SATURATION; - } else if (strcmp(intent, "Perceptual") == 0) { - cmsIntent = INTENT_PERCEPTUAL; - } - } - } + if (state != NULL) + cmsIntent = state->getCmsRenderingIntent(); if (transformIntent == cmsIntent) { return cs; } @@ -1929,18 +1923,8 @@ GfxColorSpace *GfxICCBasedColorSpace::parse(Array *arr, OutputDev *out, GfxState cmsHTRANSFORM transform; int cmsIntent = INTENT_RELATIVE_COLORIMETRIC; - if (state != NULL) { - const char *intent = state->getRenderingIntent(); - if (intent != NULL) { - if (strcmp(intent, "AbsoluteColorimetric") == 0) { - cmsIntent = INTENT_ABSOLUTE_COLORIMETRIC; - } else if (strcmp(intent, "Saturation") == 0) { - cmsIntent = INTENT_SATURATION; - } else if (strcmp(intent, "Perceptual") == 0) { - cmsIntent = INTENT_PERCEPTUAL; - } - } - } + if (state != NULL) + cmsIntent = state->getCmsRenderingIntent(); if ((transform = cmsCreateTransform(hp, COLORSPACE_SH(cst) |CHANNELS_SH(nCompsA) | BYTES_SH(1), dhp, @@ -1950,7 +1934,7 @@ GfxColorSpace *GfxICCBasedColorSpace::parse(Array *arr, OutputDev *out, GfxState error(errSyntaxWarning, -1, "Can't create transform"); cs->transform = NULL; } else { - cs->transform = new GfxColorTransform(transform, cmsIntent, dcst); + cs->transform = new GfxColorTransform(hp, transform, cmsIntent, dcst); } if (dcst == PT_RGB || dcst == PT_CMYK) { // create line transform only when the display is RGB type color space @@ -1960,10 +1944,9 @@ GfxColorSpace *GfxICCBasedColorSpace::parse(Array *arr, OutputDev *out, GfxState error(errSyntaxWarning, -1, "Can't create transform"); cs->lineTransform = NULL; } else { - cs->lineTransform = new GfxColorTransform(transform, cmsIntent, dcst); + cs->lineTransform = new GfxColorTransform(NULL, transform, cmsIntent, dcst); } } - cmsCloseProfile(hp); } obj1.free(); // put this colorSpace into cache @@ -2395,6 +2378,40 @@ void GfxICCBasedColorSpace::getDefaultRanges(double *decodeLow, #endif } +#ifdef USE_CMS +char *GfxICCBasedColorSpace::getPostScriptCSA() +{ + int size; + char *buf; + cmsHPROFILE profile = transform->getSourceProfile(); + + if (profile == NULL) { + error(errSyntaxWarning, -1, "profile is NULL"); + return NULL; + } + +#if USE_LCMS1 + size = cmsGetPostScriptCSA(profile, transform->getIntent(), NULL, 0); +#else + size = cmsGetPostScriptCSA(cmsGetProfileContextID(profile), profile, transform->getIntent(), 0, NULL, 0); +#endif + if (size == 0) { + error(errSyntaxWarning, -1, "PostScript CSA is NULL"); + return NULL; + } + + buf = (char*)gmalloc(size+1); +#if USE_LCMS1 + cmsGetPostScriptCSA(profile, transform->getIntent(), buf, size); +#else + cmsGetPostScriptCSA(cmsGetProfileContextID(profile), profile, transform->getIntent(), 0, buf, size); +#endif + buf[size] = 0; + + return buf; +} +#endif + //------------------------------------------------------------------------ // GfxIndexedColorSpace //------------------------------------------------------------------------ @@ -6668,7 +6685,7 @@ void GfxState::setDisplayProfile(cmsHPROFILE localDisplayProfileA) { INTENT_RELATIVE_COLORIMETRIC,LCMS_FLAGS)) == 0) { error(errSyntaxWarning, -1, "Can't create Lab transform"); } else { - XYZ2DisplayTransformRelCol = new GfxColorTransform(transform, INTENT_RELATIVE_COLORIMETRIC, localDisplayPixelType); + XYZ2DisplayTransformRelCol = new GfxColorTransform(XYZProfile, transform, INTENT_RELATIVE_COLORIMETRIC, localDisplayPixelType); } if ((transform = cmsCreateTransform(XYZProfile, TYPE_XYZ_DBL, localDisplayProfile, @@ -6677,7 +6694,7 @@ void GfxState::setDisplayProfile(cmsHPROFILE localDisplayProfileA) { INTENT_ABSOLUTE_COLORIMETRIC,LCMS_FLAGS)) == 0) { error(errSyntaxWarning, -1, "Can't create Lab transform"); } else { - XYZ2DisplayTransformAbsCol = new GfxColorTransform(transform, INTENT_ABSOLUTE_COLORIMETRIC, localDisplayPixelType); + XYZ2DisplayTransformAbsCol = new GfxColorTransform(XYZProfile, transform, INTENT_ABSOLUTE_COLORIMETRIC, localDisplayPixelType); } if ((transform = cmsCreateTransform(XYZProfile, TYPE_XYZ_DBL, localDisplayProfile, @@ -6686,7 +6703,7 @@ void GfxState::setDisplayProfile(cmsHPROFILE localDisplayProfileA) { INTENT_SATURATION,LCMS_FLAGS)) == 0) { error(errSyntaxWarning, -1, "Can't create Lab transform"); } else { - XYZ2DisplayTransformSat = new GfxColorTransform(transform, INTENT_SATURATION, localDisplayPixelType); + XYZ2DisplayTransformSat = new GfxColorTransform(XYZProfile, transform, INTENT_SATURATION, localDisplayPixelType); } if ((transform = cmsCreateTransform(XYZProfile, TYPE_XYZ_DBL, localDisplayProfile, @@ -6695,9 +6712,8 @@ void GfxState::setDisplayProfile(cmsHPROFILE localDisplayProfileA) { INTENT_PERCEPTUAL,LCMS_FLAGS)) == 0) { error(errSyntaxWarning, -1, "Can't create Lab transform"); } else { - XYZ2DisplayTransformPerc = new GfxColorTransform(transform, INTENT_PERCEPTUAL, localDisplayPixelType); + XYZ2DisplayTransformPerc = new GfxColorTransform(XYZProfile, transform, INTENT_PERCEPTUAL, localDisplayPixelType); } - cmsCloseProfile(XYZProfile); } } @@ -6720,6 +6736,21 @@ GfxColorTransform *GfxState::getXYZ2DisplayTransform() { return transform; } +int GfxState::getCmsRenderingIntent() { + const char *intent = getRenderingIntent(); + int cmsIntent = INTENT_RELATIVE_COLORIMETRIC; + if (intent) { + if (strcmp(intent, "AbsoluteColorimetric") == 0) { + cmsIntent = INTENT_ABSOLUTE_COLORIMETRIC; + } else if (strcmp(intent, "Saturation") == 0) { + cmsIntent = INTENT_SATURATION; + } else if (strcmp(intent, "Perceptual") == 0) { + cmsIntent = INTENT_PERCEPTUAL; + } + } + return cmsIntent; +} + #endif void GfxState::setPath(GfxPath *pathA) { diff --git a/poppler/GfxState.h b/poppler/GfxState.h index 106b2c0..98c97d0 100644 --- a/poppler/GfxState.h +++ b/poppler/GfxState.h @@ -182,14 +182,16 @@ class GfxColorTransform { public: void doTransform(void *in, void *out, unsigned int size); // transformA should be a cmsHTRANSFORM - GfxColorTransform(void *transformA, int cmsIntent, unsigned int transformPixelType); + GfxColorTransform(void *sourceProfileA, void *transformA, int cmsIntent, unsigned int transformPixelType); ~GfxColorTransform(); int getIntent() { return cmsIntent; } int getTransformPixelType() { return transformPixelType; } void ref(); unsigned int unref(); + void *getSourceProfile() { return sourceProfile; } private: GfxColorTransform() {} + void *sourceProfile; void *transform; unsigned int refCount; int cmsIntent; @@ -550,6 +552,10 @@ public: // ICCBased-specific access. GfxColorSpace *getAlt() { return alt; } + Ref getRef() { return iccProfileStream; } +#ifdef USE_CMS + char *getPostScriptCSA(); +#endif private: @@ -1540,6 +1546,7 @@ public: void setDisplayProfile(void *localDisplayProfileA); void *getDisplayProfile() { return localDisplayProfile; } GfxColorTransform *getXYZ2DisplayTransform(); + int getCmsRenderingIntent(); #endif // Add to path. diff --git a/poppler/PSOutputDev.cc b/poppler/PSOutputDev.cc index 149bb62..7b9a4e1 100644 --- a/poppler/PSOutputDev.cc +++ b/poppler/PSOutputDev.cc @@ -1105,6 +1105,7 @@ PSOutputDev::PSOutputDev(const char *fileName, PDFDoc *doc, haveTextClip = gFalse; t3String = NULL; forceRasterize = forceRasterizeA; + iccEmitted = new GooHash(gTrue); // open file or pipe if (!strcmp(fileName, "-")) { @@ -1461,6 +1462,7 @@ PSOutputDev::~PSOutputDev() { customColors = cc->next; delete cc; } + delete iccEmitted; } void PSOutputDev::writeHeader(int firstPage, int lastPage, @@ -3881,7 +3883,7 @@ void PSOutputDev::updateFillColorSpace(GfxState *state) { case psLevel2: case psLevel3: if (state->getFillColorSpace()->getMode() != csPattern) { - dumpColorSpaceL2(state->getFillColorSpace(), gTrue, gFalse, gFalse); + dumpColorSpaceL2(state, state->getFillColorSpace(), gTrue, gFalse, gFalse); writePS(" cs\n"); } break; @@ -3902,7 +3904,7 @@ void PSOutputDev::updateStrokeColorSpace(GfxState *state) { case psLevel2: case psLevel3: if (state->getStrokeColorSpace()->getMode() != csPattern) { - dumpColorSpaceL2(state->getStrokeColorSpace(), gTrue, gFalse, gFalse); + dumpColorSpaceL2(state, state->getStrokeColorSpace(), gTrue, gFalse, gFalse); writePS(" CS\n"); } break; @@ -5033,12 +5035,12 @@ void PSOutputDev::drawImageMask(GfxState *state, Object *ref, Stream *str, break; case psLevel2: case psLevel2Sep: - doImageL2(ref, NULL, invert, inlineImg, str, width, height, len, + doImageL2(state, ref, NULL, invert, inlineImg, str, width, height, len, NULL, NULL, 0, 0, gFalse); break; case psLevel3: case psLevel3Sep: - doImageL3(ref, NULL, invert, inlineImg, str, width, height, len, + doImageL3(state, ref, NULL, invert, inlineImg, str, width, height, len, NULL, NULL, 0, 0, gFalse); break; } @@ -5077,12 +5079,12 @@ void PSOutputDev::drawImage(GfxState *state, Object *ref, Stream *str, break; case psLevel2: case psLevel2Sep: - doImageL2(ref, colorMap, gFalse, inlineImg, str, + doImageL2(state, ref, colorMap, gFalse, inlineImg, str, width, height, len, maskColors, NULL, 0, 0, gFalse); break; case psLevel3: case psLevel3Sep: - doImageL3(ref, colorMap, gFalse, inlineImg, str, + doImageL3(state, ref, colorMap, gFalse, inlineImg, str, width, height, len, maskColors, NULL, 0, 0, gFalse); break; } @@ -5112,12 +5114,12 @@ void PSOutputDev::drawMaskedImage(GfxState *state, Object *ref, Stream *str, break; case psLevel2: case psLevel2Sep: - doImageL2(ref, colorMap, gFalse, gFalse, str, width, height, len, + doImageL2(state, ref, colorMap, gFalse, gFalse, str, width, height, len, NULL, maskStr, maskWidth, maskHeight, maskInvert); break; case psLevel3: case psLevel3Sep: - doImageL3(ref, colorMap, gFalse, gFalse, str, width, height, len, + doImageL3(state, ref, colorMap, gFalse, gFalse, str, width, height, len, NULL, maskStr, maskWidth, maskHeight, maskInvert); break; } @@ -5508,7 +5510,7 @@ void PSOutputDev::maskToClippingPath(Stream *maskStr, int maskWidth, int maskHei maskStr->close(); } -void PSOutputDev::doImageL2(Object *ref, GfxImageColorMap *colorMap, +void PSOutputDev::doImageL2(GfxState *state, Object *ref, GfxImageColorMap *colorMap, GBool invert, GBool inlineImg, Stream *str, int width, int height, int len, int *maskColors, Stream *maskStr, @@ -5696,7 +5698,7 @@ void PSOutputDev::doImageL2(Object *ref, GfxImageColorMap *colorMap, // color space if (colorMap) { - dumpColorSpaceL2(colorMap->getColorSpace(), gFalse, gTrue, gFalse); + dumpColorSpaceL2(state, colorMap->getColorSpace(), gFalse, gTrue, gFalse); writePS(" setcolorspace\n"); } @@ -5972,7 +5974,7 @@ void PSOutputDev::doImageL2(Object *ref, GfxImageColorMap *colorMap, } //~ this doesn't currently support OPI -void PSOutputDev::doImageL3(Object *ref, GfxImageColorMap *colorMap, +void PSOutputDev::doImageL3(GfxState *state, Object *ref, GfxImageColorMap *colorMap, GBool invert, GBool inlineImg, Stream *str, int width, int height, int len, int *maskColors, Stream *maskStr, @@ -6072,7 +6074,7 @@ void PSOutputDev::doImageL3(Object *ref, GfxImageColorMap *colorMap, // color space if (colorMap) { - dumpColorSpaceL2(colorMap->getColorSpace(), gFalse, gTrue, gFalse); + dumpColorSpaceL2(state, colorMap->getColorSpace(), gFalse, gTrue, gFalse); writePS(" setcolorspace\n"); } @@ -6345,7 +6347,7 @@ void PSOutputDev::doImageL3(Object *ref, GfxImageColorMap *colorMap, } } -void PSOutputDev::dumpColorSpaceL2(GfxColorSpace *colorSpace, +void PSOutputDev::dumpColorSpaceL2(GfxState *state, GfxColorSpace *colorSpace, GBool genXform, GBool updateColors, GBool map01) { GfxCalGrayColorSpace *calGrayCS; @@ -6487,17 +6489,48 @@ void PSOutputDev::dumpColorSpaceL2(GfxColorSpace *colorSpace, break; case csICCBased: +#if USE_CMS + { + GfxICCBasedColorSpace *iccBasedCS; + iccBasedCS = (GfxICCBasedColorSpace *)colorSpace; + Ref ref = iccBasedCS->getRef(); + int intent = state->getCmsRenderingIntent(); + GooString *name = GooString::format("ICCBased-{0:d}-{1:d}-{2:d}", ref.num, ref.gen, intent); + if (iccEmitted->lookupInt(name)) { + writePSFmt("{0:t}", name); + if (genXform) { + writePS(" {}"); + } + } else { + char *csa = iccBasedCS->getPostScriptCSA(); + if (csa) { + writePSFmt("userdict /{0:t} {1:s} put\n", name, csa); + iccEmitted->add(name->copy(), 1); + writePSFmt("{0:t}", name); + if (genXform) { + writePS(" {}"); + } + gfree(csa); + } else { + dumpColorSpaceL2(state, ((GfxICCBasedColorSpace *)colorSpace)->getAlt(), + genXform, updateColors, gFalse); + } + delete name; + } + } +#else // there is no transform function to the alternate color space, so // we can use it directly dumpColorSpaceL2(((GfxICCBasedColorSpace *)colorSpace)->getAlt(), genXform, updateColors, gFalse); +#endif break; case csIndexed: indexedCS = (GfxIndexedColorSpace *)colorSpace; baseCS = indexedCS->getBase(); writePS("[/Indexed "); - dumpColorSpaceL2(baseCS, gFalse, gFalse, gTrue); + dumpColorSpaceL2(state, baseCS, gFalse, gFalse, gTrue); n = indexedCS->getIndexHigh(); numComps = baseCS->getNComps(); lookup = indexedCS->getLookup(); @@ -6572,7 +6605,7 @@ void PSOutputDev::dumpColorSpaceL2(GfxColorSpace *colorSpace, writePS("[/Separation "); writePSString(separationCS->getName()); writePS(" "); - dumpColorSpaceL2(separationCS->getAlt(), gFalse, gFalse, gFalse); + dumpColorSpaceL2(state, separationCS->getAlt(), gFalse, gFalse, gFalse); writePS("\n"); cvtFunction(separationCS->getFunc()); writePS("]"); @@ -6594,7 +6627,7 @@ void PSOutputDev::dumpColorSpaceL2(GfxColorSpace *colorSpace, writePS(" "); } writePS("]\n"); - dumpColorSpaceL2(deviceNCS->getAlt(), gFalse, updateColors, gFalse); + dumpColorSpaceL2(state, deviceNCS->getAlt(), gFalse, updateColors, gFalse); writePS("\n"); cvtFunction(deviceNCS->getTintTransformFunc(), map01 && deviceNCS->getAlt()->getMode() == csLab); writePS("]\n"); @@ -6603,7 +6636,7 @@ void PSOutputDev::dumpColorSpaceL2(GfxColorSpace *colorSpace, } } else { // DeviceN color spaces are a Level 3 PostScript feature. - dumpColorSpaceL2(deviceNCS->getAlt(), gFalse, updateColors, map01); + dumpColorSpaceL2(state, deviceNCS->getAlt(), gFalse, updateColors, map01); if (genXform) { writePS(" "); cvtFunction(deviceNCS->getTintTransformFunc()); diff --git a/poppler/PSOutputDev.h b/poppler/PSOutputDev.h index f30204d..118ffdc 100644 --- a/poppler/PSOutputDev.h +++ b/poppler/PSOutputDev.h @@ -352,17 +352,17 @@ private: Stream *str, int width, int height, int len, int *maskColors, Stream *maskStr, int maskWidth, int maskHeight, GBool maskInvert); - void doImageL2(Object *ref, GfxImageColorMap *colorMap, + void doImageL2(GfxState *state, Object *ref, GfxImageColorMap *colorMap, GBool invert, GBool inlineImg, Stream *str, int width, int height, int len, int *maskColors, Stream *maskStr, int maskWidth, int maskHeight, GBool maskInvert); - void doImageL3(Object *ref, GfxImageColorMap *colorMap, + void doImageL3(GfxState *state, Object *ref, GfxImageColorMap *colorMap, GBool invert, GBool inlineImg, Stream *str, int width, int height, int len, int *maskColors, Stream *maskStr, int maskWidth, int maskHeight, GBool maskInvert); - void dumpColorSpaceL2(GfxColorSpace *colorSpace, + void dumpColorSpaceL2(GfxState *stae, GfxColorSpace *colorSpace, GBool genXform, GBool updateColors, GBool map01); GBool tilingPatternFillL1(GfxState *state, Catalog *cat, Object *str, @@ -487,6 +487,8 @@ private: GBool forceRasterize; // forces the page to be rasterized into a image before printing GBool displayText; // displayText + GooHash *iccEmitted; // contains ICCBased CSAs that have been emitted + #if OPI_SUPPORT int opi13Nest; // nesting level of OPI 1.3 objects int opi20Nest; // nesting level of OPI 2.0 objects -- 2.1.1