/*
* appliers.cpp -- specific OutputRule implementations
* by
[email protected] at Sat Mar 16 14:45:02 CET 2002
* Blanca JAI additions at Tue May 21 13:18:26 CEST 2002
* TIFF output at Tue Jun 4 19:50:45 CEST 2002
*/
#ifdef __GNUC__
#ifndef __clang__
#pragma implementation
#endif
#endif
#include "rule.hpp"
#include "error.hpp"
#include "encoder.hpp"
#include "in_jai.hpp"
#include "crc32.h" /* crc32() used by out_png_work() */
#include <string.h>
/** Appends 4 bytes in MSB first (network) byte order */
static inline void mf32(char *&p, slen_t u32) {
p[0]=(u32>>24)&255;
p[1]=(u32>>16)&255;
p[2]=(u32>> 8)&255;
p[3]=(u32 )&255;
p+=4;
}
/** Appends 4 bytes in LSB first (PC) byte order */
static inline void lf32(char *&p, slen_t u32) {
p[0]=(u32 )&255;
p[1]=(u32>> 8)&255;
p[2]=(u32>>16)&255;
p[3]=(u32>>24)&255;
p+=4;
}
/** Appends 2 bytes in LSB first (PC) byte order */
static inline void lf16(char *&p, unsigned u16) {
p[0]=(u16 )&255;
p[1]=(u16>> 8)&255;
p+=2;
}
/* --- at Sat Mar 23 15:42:17 CET 2002, removed obsolete
* PostScript Level2 FlateEncode or LZWEncode filter, predictors supported
* written at Mar 16 14:48:27 CET 2002
*/
/* --- Sun Mar 17 15:43:20 CET 2002 */
#if 0
/* l1fa85g.tte was a special .tte that is not built in to bts.ttt, but I've
* integrated it into bts2.ttt at Sun Sep 22 00:48:21 CEST 2002. I've also
* integrated _l1fa85g_ to _l1c_ that day.
*/
static char *l1fa85g_tte=
#include "l1fa85g.tth"
/** PostScript Level1 FlateEncode, A85, without Predictor */
Rule::Applier::cons_t out_l1fa85g_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (!cache->isPS()
|| cache->Compression!=Rule::Cache::CO_ZIP
|| cache->hasPredictor()
|| !cache->isGray()
|| cache->TransferEncoding!=cache->TE_A85
|| (!cache->WarningOK && !cache->isPSL3())
) return Rule::Applier::DONT_KNOW;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_l1fa85g_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
/* by
[email protected] at Sun Mar 17 15:52:48 CET 2002 */
// Imp: two other TransferEncodings [no more, since #if 0]
if (out_l1fa85g_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
if (!or_->cache.isPSL3())
Error::sev(Error::WARNING_DEFER) << "l1fa85g: /ZIP without /PSL3 will be slow" << (Error*)0;
or_->doSampleFormat(sf);
PSEncoder *bp=PSEncoder::newASCII85Encode(out, or_->cacheHints.TransferCPL);
PSEncoder *cp=PSEncoder::newFlateEncode(*bp, or_->cacheHints.Effort);
Rule::writeTTE(out, out, *cp, l1fa85g_tte, or_, sf, Rule::writeData);
delete bp;
delete cp;
return Rule::Applier::OK;
}
Rule::Applier out_l1fa85g_applier = { "l1fa85g", out_l1fa85g_check_rule, out_l1fa85g_work, 0 };
#endif
/* l2jbin --- Sun Mar 17 21:45:22 CET 2002
* p0jbin (integrated to p0jbin) --- Mon Apr 15 23:29:13 CEST 2002
*/
//static char *l2jbin_tte=
//#include "l2jbin.tth"
/** PostScript Level2 DCTEncode (Baseline JPEG), Binary */
Rule::Applier::cons_t out_l2jbin_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if ((!cache->isPS() && !cache->isPDF())
|| cache->Compression!=Rule::Cache::CO_JAI
) return Rule::Applier::DONT_KNOW;
/* Dat: not unrequired anymore: cache->TransferEncoding!=cache->TE_Binary */
bool badp=false;
if (cache->isPS() && !cache->isPSL2()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/PSL* /Compression/JAI requires /PSL2+" << (Error*)0;
badp=true;
}
if (cache->SampleFormat!=Image::SF_Asis) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/PSL*|PDF* /Compression/JAI requires /SampleFormat/Asis" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/PSL*|PDF* /Compression/JAI requires /Predictor 1" << (Error*)0;
badp=true;
}
return (Rule::Applier::cons_t)(badp ? 0+Rule::Applier::BAD : 0+Rule::Applier::OK);
/* ^^^ 0+: pacify g++-3.1 */
}
Rule::Applier::cons_t out_l2jbin_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
char const*strings[]={ (char const*)NULLP/*LanguageLevel, PDF-1.`0'*/, ""/*colorSpace*/ };
//, " F closefile T closefile"/*closes*/ };
// Error::sev(Error::WARNING_DEFER) << "l2jbin: /ZIP without /PSL3 will be slow" << (Error*)0;
if (out_l2jbin_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
assert(sf->getImg()->getBpc()==8);
or_->doSampleFormat(sf);
// PSEncoder *bp=PSEncoder::newASCII85Encode(out, or_->cacheHints.TransferCPL);
Rule::Cache *cache=&or_->cache;
Filter::VerbatimE outve(out); /* required since template /p0jbin is TTM */
GenBuffer::Writable *op=cache->isPDF() ? &outve : &out, *tp=op;
strings[0]=const_cast<char*>(cache->isPDF() ? (char const*)"0" : (char const*)"2");
if (cache->TransferEncoding==cache->TE_A85) tp=PSEncoder::newASCII85Encode (*op, or_->cacheHints.TransferCPL);
else if (cache->TransferEncoding==cache->TE_Hex) tp=PSEncoder::newASCIIHexEncode(*op, or_->cacheHints.TransferCPL);
// else strings[2]=" F closefile";
Rule::writeTTT(*op, *tp, *tp, !cache->isPDF()?"l2jbin":cache->isPDFB()?"p0jbb":"p0jbin", or_, sf, Rule::writePalData, strings);
if (tp!=op) delete tp;
/* Dat: outve is deleted by C++ scope */
// if (tp!=op) delete tp; /* BUGFIX at Thu Jan 20 15:04:47 CET 2005 */
return Rule::Applier::OK;
}
Rule::Applier out_l2jbin_applier = { "PSL2+PDF-JAI", out_l2jbin_check_rule, out_l2jbin_work, 0 };
/* --- */
#if 0 /* integrated to _l2jbin_ at Sun Sep 22 14:29:13 CEST 2002 */
//static char *p0jbin_ttm=
//#include "p0jbin.tth"
/** PostScript Level2 DCTEncode (Baseline JPEG), Binary */
Rule::Applier::cons_t out_p0jbin_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (!cache->isPDF()
|| cache->Compression!=Rule::Cache::CO_JAI
) return Rule::Applier::DONT_KNOW;
/* Dat: not unrequired anymore: cache->TransferEncoding!=cache->TE_Binary */
bool badp=false;
if (cache->SampleFormat!=Image::SF_Asis) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/PDF* /Compression/JAI requires /SampleFormat/Asis" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/PDF* /Compression/JAI requires /Predictor 1" << (Error*)0;
badp=true;
}
if (badp) return Rule::Applier::BAD;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_p0jbin_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
char *strings[]={ "0"/*PDF-1.`0'*/, ""/*colorSpace*/ };
// " F closefile"/*closes*/ };
if (out_p0jbin_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
assert(sf->getImg()->getBpc()==8);
or_->doSampleFormat(sf);
Rule::Cache *cache=&or_->cache;
Filter::VerbatimE outve(out); /* required since template /p0jbin is TTM */
GenBuffer::Writable *tp=&outve;
if (cache->TransferEncoding==cache->TE_A85) tp=PSEncoder::newASCII85Encode (outve, or_->cacheHints.TransferCPL);
else if (cache->TransferEncoding==cache->TE_Hex) tp=PSEncoder::newASCIIHexEncode(outve, or_->cacheHints.TransferCPL);
else strings[2]="";
#if 0 /* old */
MiniPS::VALUE ttm;
{ Filter::FlatR flatD(p0jbin_ttm);
MiniPS::Parser p(&flatD);
ttm=p.parse1();
if (p.parse1(p.EOF_ALLOWED)!=MiniPS::Qundef || MiniPS::getType(ttm)!=MiniPS::T_ARRAY)
Error::sev(Error::EERROR) << "TTM: the TTM file should contain a single array" << (Error*)0;
/* ^^^ Dat: the result of the second p.parse1() doesn't get delete0(...)d */
}
Filter::VerbatimE outve(out);
Rule::writeTTM(outve, outve, outve, MiniPS::RARRAY(ttm), or_, sf, Rule::writePalData, strings);
MiniPS::delete0(ttm);
#else /* new */
Rule::writeTTT(outve, *tp, *tp, cache->isPDFB()?"p0jbb":"p0jbin", or_, sf, Rule::writePalData, strings);
#endif
if (tp!=&outve)delete tp;
return Rule::Applier::OK;
}
Rule::Applier out_p0jbin_applier = { "PDF-JAI", out_p0jbin_check_rule, out_p0jbin_work, 0 };
#endif
/* --- Fri Mar 22 11:52:53 CET 2002 */
/* PDF added at Sat Apr 20 20:07:34 CEST 2002 */
/* Integrated l23ind1 at Sat Apr 20 20:24:21 CEST 2002 */
//static char *l23_tte=
//#include "l23.tth"
/** PostScript Level2 and PDF generic non-transparent */
Rule::Applier::cons_t out_l23_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
unsigned char sf=cache->SampleFormat;
if (cache->hasPredictor() && cache->Compression!=Rule::Cache::CO_ZIP && cache->Compression!=Rule::Cache::CO_LZW) {
Error::sev(Error::WARNING_DEFER) << "check_rule: real /Predictor requires /ZIP or /LZW" << (Error*)0;
return Rule::Applier::BAD;
}
// assert(cache->isPSL2() && (sf==Image::SF_Transparent2 || sf==Image::SF_Transparent4 || sf==Image::SF_Transparent8));
if ((!cache->isPSL2() && !cache->isPDF())
|| (sf!=Image::SF_Gray1 && sf!=Image::SF_Gray2 && sf!=Image::SF_Gray4 && sf!=Image::SF_Gray8
&& sf!=Image::SF_Indexed1 && sf!=Image::SF_Indexed2 && sf!=Image::SF_Indexed4 && sf!=Image::SF_Indexed8
&& sf!=Image::SF_Rgb1 && sf!=Image::SF_Rgb2 && sf!=Image::SF_Rgb4 && sf!=Image::SF_Rgb8
&& sf!=Image::SF_Transparent2 && sf!=Image::SF_Transparent4 && sf!=Image::SF_Transparent8
&& sf!=Image::SF_Mask)
|| cache->TransferEncoding==cache->TE_ASCII
|| !cache->isZIPOK()
) return Rule::Applier::DONT_KNOW;
#if !HAVE_LZW
if (cache->Compression==Rule::Cache::CO_LZW) return Rule::Applier::DONT_KNOW;
#endif
// if (cache->isDCTE() && !cache->isRGB() && !cache->isGray()) {
bool badp=false;
if (cache->isDCTE() && cache->SampleFormat!=Image::SF_Rgb8 && cache->SampleFormat!=Image::SF_Gray8 && cache->SampleFormat!=Image::SF_Indexed8) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /DCTEncode requires /Rgb8 or /Gray8 (or /Indexed8)" << (Error*)0;
badp=true;
}
if (sf==Image::SF_Transparent2 || sf==Image::SF_Transparent4 || sf==Image::SF_Transparent8) {
if (cache->isPDF()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: unsupported /Transparent+ for /PDF*" << (Error*)0;
badp=true;
} else assert(cache->isPSL2());
}
return (Rule::Applier::cons_t)(badp ? 0+Rule::Applier::BAD : 0+Rule::Applier::OK);
}
Rule::Applier::cons_t out_l23_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
Rule::stream_writer_t writeXData=Rule::writePalData;
char LanguageLevel[2]="2", closes[30];
SimBuffer::B colorSpace;
char const*strings[]={ LanguageLevel, (char*)NULLP /*colorSpace*/, closes };
Rule::Cache *cache=&or_->cache;
// assert(0);
if (out_l23_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
or_->doSampleFormat(sf, true);
if (cache->Compression==Rule::Cache::CO_ZIP) {
if (!cache->isPDF()) LanguageLevel[0]='3';
} else if (cache->isPDF()) LanguageLevel[0]='0';
if (cache->isIndexed() || cache->isTransparentM()) {
unsigned ncols=PTS_dynamic_cast(Image::Indexed*,sf->getImg())->getNcols();
colorSpace << "[/Indexed/DeviceRGB " << ncols-1;
if (ncols==0) { /* Avoid writing zero bytes to the filters */
/* Imp: verify this in Acrobat Reader */
colorSpace << "()]";
} else if (!cache->isPDF()) { /* Insert an in-line hexdump. No better way :-( */
/*sprintf(colorSpace, "[/Indexed/DeviceRGB %u T %u string readstring pop]", ncols-1, ncols*3); */
colorSpace << " T " << ncols*3 << " string readstring pop]";
} else { /* Insert an in-line hexdump. No shorter way for PDF */
#if 0
colorSpace="/DeviceGray ";
#else
/* SUXX: gs5.50 & PDF & /Indexed doesn't work */
colorSpace << "\n<";
GenBuffer::Writable *hp=PSEncoder::newASCIIHexEncode(colorSpace, or_->cacheHints.TransferCPL);
hp->vi_write(sf->getImg()->getHeadp(), ncols*3);
hp->vi_write(0,0);
colorSpace << ']'; /* Dat: '>' is appended by ASCIIHexEncode */
#endif
writeXData=Rule::writeData; /* The palette has already been written. */
}
} else if (cache->isGray()) colorSpace="/DeviceGray ";
else { assert(cache->isRGB()); colorSpace="/DeviceRGB "; }
if (cache->SampleFormat==Image::SF_Mask || cache->SampleFormat==Image::SF_Indexed1) writeXData=Rule::writeData;
GenBuffer::Writable *vp=&out;
if (cache->isPDF()) vp=new Filter::VerbatimE(out); /* required since template /p02* is TTM */
GenBuffer::Writable *tp=vp;
if (cache->TransferEncoding==cache->TE_A85) tp=PSEncoder::newASCII85Encode (*vp, or_->cacheHints.TransferCPL);
else if (cache->TransferEncoding==cache->TE_Hex) tp=PSEncoder::newASCIIHexEncode(*vp, or_->cacheHints.TransferCPL);
GenBuffer::Writable *cp=tp;
switch (cache->Compression) {
case Rule::Cache::CO_None: break;
case Rule::Cache::CO_ZIP: cp=PSEncoder::newFlateEncode(*tp, or_->cacheHints.Effort); break;
case Rule::Cache::CO_LZW: cp=PSEncoder::newLZWEncode(*tp); break;
case Rule::Cache::CO_RLE: cp=PSEncoder::newRunLengthEncode(*tp, or_->cacheHints.RecordSize); break;
case Rule::Cache::CO_Fax: cp=PSEncoder::newCCITTFaxEncode(*tp, or_->cacheHints.K, or_->cacheHints.EncoderBPL, /*EndOfLine:*/ or_->cacheHints.K>0); break;
/* ^^^ getBpp() BUGFIX at Wed Jul 3 20:00:30 CEST 2002 */
/* ^^^ EndOfLine BUGFIX at Wed Jul 3 21:12:54 CEST 2002
* With EndOfLine==false, `sam2p -c:fax:1', acroread triggers the bug.
* With EndOfLine==false, `sam2p -c:fax:2', acroread and gs trigger the bug.
*/
case Rule::Cache::CO_IJG: cp=PSEncoder::newDCTIJGEncode(*tp, or_->cacheHints.EncoderColumns, or_->cacheHints.EncoderRows, or_->cacheHints.EncoderColors, or_->cacheHints.Quality); break;
case Rule::Cache::CO_DCT: { SimBuffer::B other_parameters;
or_->cacheHints.DCT->dump(other_parameters, 0, false);
cp=PSEncoder::newDCTEncode(*tp, or_->cacheHints.EncoderColumns, or_->cacheHints.EncoderRows, or_->cacheHints.EncoderColors, or_->cacheHints.ColorTransform, other_parameters);
break; }
default: assert(0);
}
GenBuffer::Writable *pp=cp;
if (cache->hasPredictor()) pp=PSEncoder::newPredictor(*cp, cache->Predictor, or_->cacheHints.PredictorBPC, or_->cacheHints.PredictorColumns, or_->cacheHints.PredictorColors);
#if 0 /* Sun Sep 22 20:40:51 CEST 2002 */
if (cp!=tp) strcpy(closes," F closefile"); else closes[0]='\0';
if (tp!=vp) strcpy(closes+strlen(closes)," T closefile");
#endif
strings[1]=colorSpace.term0()();
Rule::writeTTT(*vp, *tp, *pp,
!cache->isPDF() ? (
cache->SampleFormat==Image::SF_Indexed1 ? "l23ind1" :
cache->SampleFormat==Image::SF_Mask ? "l23mask" :
cache->isTransparentM() ? "l23tran2" :
"l23"
) : cache->isPDFB() ? (
cache->SampleFormat==Image::SF_Indexed1 ? "p02ind1bb" :
cache->SampleFormat==Image::SF_Mask ? "p02maskbb" :
"p02bb"
) : (
cache->SampleFormat==Image::SF_Indexed1 ? "p02ind1" :
cache->SampleFormat==Image::SF_Mask ? "p02mask" :
"p02"
), or_, sf, writeXData, strings);
if (pp!=cp) delete pp;
if (cp!=tp) delete cp;
if (tp!=vp) delete tp;
if (vp!=&out)delete vp;
return Rule::Applier::OK;
}
Rule::Applier out_l23_applier = { "PSL23+PDF", out_l23_check_rule, out_l23_work, 0 };
/* --- Fri Mar 22 17:22:40 CET 2002 -- Sat Jun 15 16:03:06 CEST 2002 */
/* integrated l1tr at Sat Jun 15 16:03:03 CEST 2002 */
/* added /Bbox at Sat Jun 15 16:03:31 CEST 2002 */
//static char *l1tr_tte=
//#include "l1tr.tth"
/** PostScript Level1 or PDF Fully transparent image */
Rule::Applier::cons_t out_l1tr_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (!(cache->isPS() || cache->isPDF())
|| (cache->SampleFormat!=Image::SF_Transparent && cache->SampleFormat!=Image::SF_Bbox && cache->SampleFormat!=Image::SF_Opaque)
) return Rule::Applier::DONT_KNOW;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_l1tr_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
char t[]="....";
if (out_l1tr_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
or_->doSampleFormat(sf);
Rule::Cache *cache=&or_->cache;
Filter::VerbatimE outve(out); /* required since template /p0* is TTM */
if (cache->isPS()) { t[0]='l'; t[1]='1'; }
else { t[0]='p'; t[1]='0'; }
switch (cache->SampleFormat) {
case Image::SF_Transparent: t[2]='t'; t[3]='r'; break;
case Image::SF_Opaque: t[2]='o'; t[3]='p'; break;
case Image::SF_Bbox: t[2]='b'; t[3]='b'; break;
default: assert(0);
}
Rule::writeTTT(outve, outve, outve, t, or_, sf, 0/*NULLP*/);
return Rule::Applier::OK;
}
Rule::Applier out_l1tr_applier = { "P-TrOpBb", out_l1tr_check_rule, out_l1tr_work, 0 };
/* --- */
/* --- l23mask Fri Mar 22 18:11:12 CET 2002
* --- l23ind1 Fri Mar 22 18:11:12 CET 2002
* removed (integrated to l23) at Sat Apr 20 20:25:57 CEST 2002
*/
/* --- l1mask Fri Mar 22 18:33:01 CET 2002
* --- l1mashex Sun Apr 14 15:25:25 CEST 2002
* --- l1in1 Fri Mar 22 18:33:37 CET 2002
* --- l1in1hex Fri Mar 22 18:33:37 CET 2002
* removed (integrated to l1c) at Sat Apr 20 20:25:57 CEST 2002
* --- lcr Sat Jun 1 17:09:57 CEST 2002
* removed (integrated to l1c) at Sun Jun 2 16:48:31 CEST 2002
* --- l1gbh
* removed (integrated to l1c) at Sun Jun 2 16:48:31 CEST 2002
* --- l1fa85g.tte: PostScript Level1 FlateEncode, A85, without Predictor
* removed (integrated to l1c) at Sun Sep 22 00:48:21 CEST 2002
*/
/* --- Sun Jun 2 16:48:44 CEST 2002 */
//static char *l1mask_tte=
//#include "l1mask.tth"
static void gen_tkey(char *tkey, GenBuffer::Writable& out, GenBuffer::Writable*&tp, GenBuffer::Writable*& cp, Rule::OutputRule*or_) {
Rule::Cache *cache=&or_->cache;
tp=&out;
if (cache->TransferEncoding==cache->TE_A85) { tkey[3]='8'; tp=PSEncoder::newASCII85Encode (out, or_->cacheHints.TransferCPL); }
else if (cache->TransferEncoding==cache->TE_Hex) { tkey[3]='h'; tp=PSEncoder::newASCIIHexEncode(out, or_->cacheHints.TransferCPL); }
else tkey[3]='b';
cp=tp;
if (cache->Compression==Rule::Cache::CO_RLE) { tkey[4]='r'; cp=PSEncoder::newRunLengthEncode(*tp, or_->cacheHints.RecordSize); }
else if (cache->Compression==Rule::Cache::CO_ZIP) { tkey[4]='z'; cp=PSEncoder::newFlateEncode(*tp, or_->cacheHints.Effort); }
else if (cache->Compression==Rule::Cache::CO_LZW) { tkey[4]='l'; cp=PSEncoder::newLZWEncode(*tp); }
else tkey[4]='n';
/* vvv removed 'm' and '1' at Sun Sep 22 17:53:08 CEST 2002 */
tkey[2]=// cache->SampleFormat==Image::SF_Mask ? 'm' :
// cache->SampleFormat==Image::SF_Indexed1 ? '1' :
cache->SampleFormat==Image::SF_Indexed2 ? '2' :
cache->SampleFormat==Image::SF_Indexed4 ? '4' :
cache->SampleFormat==Image::SF_Indexed8 ? '8' :
cache->SampleFormat==Image::SF_Transparent2 ? 't' :
cache->SampleFormat==Image::SF_Transparent4 ? 't' :
cache->SampleFormat==Image::SF_Transparent8 ? 't' :
'g'; /* /Gray*, /Rgb*, /Mask, /Indexed1 */
}
/** PostScript Level1 uncompressed binary or hex */
Rule::Applier::cons_t out_l1c_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if ((cache->FileFormat!=cache->FF_PSL1 && cache->FileFormat!=cache->FF_PSLC && cache->FileFormat!=cache->FF_PSL2)
|| (cache->FileFormat==cache->FF_PSL2 && cache->Compression!=Rule::Cache::CO_ZIP)
|| (cache->TransferEncoding!=cache->TE_Binary && cache->TransferEncoding!=cache->TE_Hex && cache->TransferEncoding!=cache->TE_A85)
|| (cache->Compression!=Rule::Cache::CO_None && cache->Compression!=Rule::Cache::CO_RLE && cache->Compression!=Rule::Cache::CO_ZIP && cache->Compression!=Rule::Cache::CO_LZW)
|| cache->hasPredictor()
|| !(cache->isTransparentM() || cache->isIndexed() || cache->isGray() || cache->isRGB())
) return Rule::Applier::DONT_KNOW;
bool badp=false;
if (cache->FileFormat==cache->FF_PSL1 && cache->SampleFormat!=Image::SF_Indexed1 && cache->isIndexed()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /SampleFormat/Indexed+ doesn't work with /FileFormat/PSL1 (use /PSLC or /PSL2)" << (Error*)0;
badp=true;
}
if (cache->FileFormat==cache->FF_PSL1 && cache->isRGB()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /SampleFormat/RGB* doesn't work with /FileFormat/PSL1 (use /PSLC or /PSL2)" << (Error*)0;
badp=true;
}
char tkey[]="l1..."; /* /l1{2,4,8,t}{8,h}{r,z,l} */
GenBuffer::Writable *tp0,*cp0,*out=(GenBuffer::Writable*)NULLP;
gen_tkey(tkey, *out, tp0, cp0, or_);
/* fprintf(stderr,"tkey=%s\n", tkey); */
if (!badp && Rule::Templates->get(tkey, strlen(tkey))==MiniPS::Qundef) return Rule::Applier::DONT_KNOW;
return (Rule::Applier::cons_t)(badp ? 0+Rule::Applier::BAD : 0+Rule::Applier::OK);
/* ^^^ Dat: 0+: pacify gcc-3.1 */
}
Rule::Applier::cons_t out_l1c_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
if (out_l1c_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
/* Dat: only these have been defined so far: grep '^/l1[248tg][8h][rzl]' bts2.ttt
* /l1thr /l1g8r /l1ghr /l128r /l12hr /l148r /l14hr /l188r /l18hr /l1g8z /l1ghz /l1g8l /l1ghl
*/
or_->doSampleFormat(sf, true); /* Dat: `true': because of /Transparent+ */
char tkey[]="l1..."; /* /l1{2,4,8,t}{8,h}{r,z,l} */
GenBuffer::Writable *tp, *cp;
gen_tkey(tkey, out, tp, cp, or_);
// fprintf(stderr, "tkey=(%s)\n", tkey);
Rule::writeTTT(out, *tp, *cp, tkey, or_, sf,
tkey[2]=='2' || tkey[2]=='4' || tkey[2]=='8' || tkey[2]=='t' ? Rule::writePalData : Rule::writeData
);
// Rule::writeTTT(out, *tp, *cp, tkey, or_, sf, Rule::writePalData);
if (cp!=tp) delete cp;
if (tp!=&out)delete tp;
return Rule::Applier::OK;
}
Rule::Applier out_l1c_applier = { "PSL1C", out_l1c_check_rule, out_l1c_work, 0 };
/* lcrbin (lcrb) --- Sun Apr 14 16:50:14 CEST 2002
* lcrhex (lcr8, lcrh) --- Sun Apr 14 16:50:22 CEST 2002
* removed (integrated to _l1c_) at Sat Jun 1 17:09:52 CEST 2002
*/
#if 0
/* --- Sat Jun 1 17:09:57 CEST 2002 */
//static char *lcrbin_tte=
//#include "lcrbin.tth"
/** PostScript Level1+C uncompressed RGB image */
Rule::Applier::cons_t out_lcr_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (!cache->isPS()
|| cache->FileFormat==cache->FF_PSL1
|| (cache->TransferEncoding!=cache->TE_Binary && cache->TransferEncoding!=cache->TE_Hex && cache->TransferEncoding!=cache->TE_A85)
|| cache->Compression!=Rule::Cache::CO_None
|| cache->hasPredictor()
|| !cache->isRGB()
) return Rule::Applier::DONT_KNOW;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_lcr_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
Rule::Cache *cache=&or_->cache;
if (out_lcr_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
or_->doSampleFormat(sf);
char tkey[]="lcr.";
GenBuffer::Writable *tp=&out;
if (cache->TransferEncoding==cache->TE_A85) { tkey[3]='8'; tp=PSEncoder::newASCII85Encode (out, or_->cacheHints.TransferCPL); }
else if (cache->TransferEncoding==cache->TE_Hex) { tkey[3]='h'; tp=PSEncoder::newASCIIHexEncode(out, or_->cacheHints.TransferCPL); }
else tkey[3]='b';
Rule::writeTTT(out, *tp, *tp, tkey, or_, sf, Rule::writeData);
if (tp!=&out)delete tp;
return Rule::Applier::OK;
}
Rule::Applier out_lcr_applier = { "PSLC-RGB", out_lcr_check_rule, out_lcr_work, 0 };
#endif
/* --- Sat Mar 23 12:37:04 CET 2002; Tue Jul 2 10:23:44 CEST 2002 */
Rule::Applier::cons_t out_gif89a_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
bool badp=false;
if (cache->FileFormat!=cache->FF_GIF89a
) return Rule::Applier::DONT_KNOW;
if (!cache->isIndexed() && !cache->isTransparentM()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /GIF89a must be /Indexed*, /Mask or /Transparent+" << (Error*)0;
badp=true;
}
if (cache->TransferEncoding!=cache->TE_Binary) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /GIF89a requires /Binary" << (Error*)0;
badp=true;
}
if (cache->Compression!=Rule::Cache::CO_None && cache->Compression!=Rule::Cache::CO_LZW) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /GIF89a requires /LZW" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /GIF89a requires /Predictor 1" << (Error*)0;
badp=true;
}
#if !USE_OUT_GIF
Error::sev(Error::WARNING_DEFER) << "check_rule: please `configure --enable-gif' for /GIF89a" << (Error*)0;
badp=true;
#endif
if (badp) return Rule::Applier::BAD;
or_->cache.WarningOK=true;
return Rule::Applier::OK;
}
#if USE_OUT_GIF
#if OBJDEP
# warning REQUIRES: out_gif.o
#endif
extern void out_gif_write(GenBuffer::Writable& out, Image::Indexed *img); /* out_gif.cpp */
Rule::Applier::cons_t out_gif89a_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
if (out_gif89a_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
or_->cache.SampleFormat=(Image::sf_t)(or_->cache.isIndexed()? 0+Image::SF_Indexed8: 0+Image::SF_Transparent8);
/* ^^^ Dat: 0+: pacify gcc-3.1; (Image::sf_t): pacify VC6.0 */
or_->doSampleFormat(sf);
out_gif_write(out, PTS_dynamic_cast(Image::Indexed*,sf->getImg()));
return Rule::Applier::OK;
}
#else
/*# define out_gif89a_check_rule (Rule::Applier::check_rule_t)NULLP*/
# define out_gif89a_work (Rule::Applier::work_t)0
#endif
Rule::Applier out_gif89a_applier = {
#if HAVE_LZW
"GIF89a+LZW"
#else
"GIF89a"
#endif
, out_gif89a_check_rule, out_gif89a_work, 0 };
/* --- Tue Jun 4 19:51:03 CEST 2002 */
Rule::Applier::cons_t out_xpm_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
bool badp=false;
if (cache->FileFormat!=cache->FF_XPM
) return Rule::Applier::DONT_KNOW;
if (!cache->isIndexed() && !cache->isTransparentM()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /XPM must be /Indexed*, /Mask or /Transparent+" << (Error*)0;
badp=true;
}
if (cache->TransferEncoding!=cache->TE_ASCII && cache->TransferEncoding!=cache->TE_Binary) {
/* ^^^ && BUGFIX at Thu Jul 11 21:57:09 CEST 2002 */
Error::sev(Error::WARNING_DEFER) << "check_rule: /XPM requires /TransferEncoding/ASCII" << (Error*)0;
badp=true;
}
if (cache->Compression!=Rule::Cache::CO_None) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /XPM requires /Compression/None" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /XPM requires /Predictor 1" << (Error*)0;
badp=true;
}
if (badp) return Rule::Applier::BAD;
or_->cache.WarningOK=true;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_xpm_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
if (out_xpm_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
or_->cache.SampleFormat=(Image::sf_t)(or_->cache.isIndexed()? 0+Image::SF_Indexed8: 0+Image::SF_Transparent8);
/* ^^^ force 8-bit; may trigger warnings... */
/* ^^^ Dat: 0+: pacify gcc-3.1 */
or_->doSampleFormat(sf);
Image::Indexed *iimg=PTS_dynamic_cast(Image::Indexed*,sf->getImg());
/* vvv 93 useful ASCII chars (+ '\0'), missing: " and \ ; plus a hextable */
static char const xpmc=93;
static char const xpms[xpmc+1]="0123456789abcdef !#$%&'()*+,-./:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[]^_`ghijklmnopqrstuvwxyz{|}~";
Image::Sampled::dimen_t wd=iimg->getWd(), htc=iimg->getHt();
bool pal2ch=iimg->getNcols()>xpmc;
// pal2ch=true;
out << "/* XPM */\nstatic char *sam2p_xpm[] = {\n"
"/* columns rows colors chars-per-pixel */\n\""
<< wd << ' ' << htc << ' ' << iimg->getNcols()
<< ' ' << (pal2ch?2:1) << "\",\n";
char const *p=iimg->getHeadp(), *pend=iimg->getRowbeg(), *phend;
char coline3[]="\".. c #ABCDEF\",\n";
char cotran3[]="\".. c None s None \",\n";
assert((pend-p)%3==0);
unsigned i=0;
bool transp=false;
if (iimg->getTransp()>=0) { pend-=3; assert((iimg->getTransp())*3+p==pend); transp=true; }
if (pal2ch) {
while (p!=pend) {
coline3[ 1]=xpms[i>>2];
coline3[ 2]=xpms[i++&3];
coline3[ 7]=xpms[*(unsigned char const*)p>>4];
coline3[ 8]=xpms[*(unsigned char const*)p++&15];
coline3[ 9]=xpms[*(unsigned char const*)p>>4];
coline3[10]=xpms[*(unsigned char const*)p++&15];
coline3[11]=xpms[*(unsigned char const*)p>>4];
coline3[12]=xpms[*(unsigned char const*)p++&15];
out << coline3;
}
if (transp) {
cotran3[ 1]=xpms[i>>2];
cotran3[ 2]=xpms[i++&3];
out << cotran3;
p+=3;
}
pend=iimg->getRowbeg()+wd*htc;
out << "/* Pixels */\n";
char *obuf=new char[4+2*wd], *op;
obuf[0]='"';
obuf[2*wd+1]='"';
obuf[2*wd+2]=','; /* Dat: it's OK to have a comma in the last line */
obuf[2*wd+3]='\n';
while (htc--!=0) {
phend=p+wd;
op=obuf;
while (p!=phend) { *++op=xpms[*p>>2]; *++op=xpms[*p++&3]; }
out.vi_write(obuf, 2*wd+4);
}
delete [] obuf;
} else {
coline3[1]='"';
while (p!=pend) {
coline3[ 2]=xpms[i++];
coline3[ 7]=xpms[*(unsigned char const*)p>>4];
coline3[ 8]=xpms[*(unsigned char const*)p++&15];
coline3[ 9]=xpms[*(unsigned char const*)p>>4];
coline3[10]=xpms[*(unsigned char const*)p++&15];
coline3[11]=xpms[*(unsigned char const*)p>>4];
coline3[12]=xpms[*(unsigned char const*)p++&15];
out << (coline3+1);
}
if (transp) {
cotran3[1]='"';
cotran3[ 2]=xpms[i];
out << (cotran3+1);
p+=3;
}
pend=iimg->getRowbeg()+wd*htc;
out << "/* Pixels */\n";
char *obuf=new char[4+wd], *op;
obuf[0]='"';
obuf[wd+1]='"';
obuf[wd+2]=','; /* Dat: it's OK to have a comma in the last line */
obuf[wd+3]='\n';
while (htc--!=0) {
phend=p+wd;
op=obuf;
while (p!=phend) *++op=xpms[0U+*p++];
out.vi_write(obuf, wd+4);
}
delete [] obuf;
}
assert(p==pend);
out << "};\n";
return Rule::Applier::OK;
}
Rule::Applier out_xpm_applier = { "XPM", out_xpm_check_rule, out_xpm_work, 0 };
/* --- Sat Mar 23 13:18:07 CET 2002 */
Rule::Applier::cons_t out_pnm_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_PNM
) return Rule::Applier::DONT_KNOW;
bool badp=false;
// if (cache->SampleFormat!=Image::SF_Rgb8 && cache->SampleFormat!=Image::SF_Gray8 && cache->SampleFormat!=Image::SF_Gray1) {
if (!cache->isGray() && !cache->isRGB() && !cache->isIndexed() && !cache->isTransparentM()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /PNM must be /Rgb8, /Gray8 or /Gray1" << (Error*)0;
badp=true;
}
if (cache->TransferEncoding!=cache->TE_Binary && cache->TransferEncoding!=cache->TE_ASCII) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /PNM requires /Binary or /ASCII" << (Error*)0;
badp=true;
}
if (cache->Compression!=Rule::Cache::CO_None) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /PNM requires /Compression/None" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /PNM requires /Predictor 1" << (Error*)0;
badp=true;
}
if (badp) return Rule::Applier::BAD;
or_->cache.WarningOK=true;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_pnm_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
/*static*/ char head[]={ 'P', 0, '\n', '#', ' ', 'b', 'y', ' ' };
/*static*/ char tmp[72];
Image::sf_t sfo=or_->cache.SampleFormat;
Image::Indexed *alphaChannel=(Image::Indexed*)NULLP;
if (out_pnm_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
if (or_->cache.SampleFormat==Image::SF_Gray1 || or_->cache.SampleFormat==Image::SF_Gray8) {
if (or_->cache.TransferEncoding==or_->cache.TE_ASCII) or_->cache.SampleFormat=Image::SF_Gray8;
} else if (or_->cache.SampleFormat==Image::SF_Gray2 || or_->cache.SampleFormat==Image::SF_Gray4) {
or_->cache.SampleFormat=Image::SF_Gray8;
} else if (or_->cache.isRGB()) {
or_->cache.SampleFormat=Image::SF_Rgb8;
} else {
assert(or_->cache.isIndexed() || or_->cache.isTransparentM());
if (or_->cache.isTransparentM()) {
alphaChannel=PTS_dynamic_cast(Image::Indexed*,sf->getImg())->calcAlpha();
sf->clearTransp(); /* BUGFIX at Tue Sep 17 10:05:47 CEST 2002 */
}
or_->cache.SampleFormat=(Image::sf_t)(
!sf->canGrayy() ? 0+Image::SF_Rgb8
: sf->minRGBBpcc()==1 ? 0+Image::SF_Gray1 : 0+Image::SF_Gray8);
/* ^^^ Dat: 0+: pacify gcc-3.1 */
}
if (or_->cache.SampleFormat==Image::SF_Gray1 && or_->cache.TransferEncoding==or_->cache.TE_ASCII)
or_->cache.SampleFormat=Image::SF_Gray8;
or_->doSampleFormat(sf);
sfo=or_->cache.SampleFormat;
head[1]=(sfo==Image::SF_Rgb8 ? '3' : sfo==Image::SF_Gray8 ? '2' : '1')
+(or_->cache.TransferEncoding==or_->cache.TE_Binary)*3;
out.vi_write(head, sizeof(head));
out << Error::banner0;
Image::Sampled *img=sf->getImg();
out << '\n' << img->getWd() << ' ' << img->getHt();
out << &(" 255\n"[sfo==Image::SF_Gray1?4:0]);
/* ^^^ SF_Gray1 BUGFIX at Tue Jun 4 21:44:17 CEST 2002 */
register char *p=img->getRowbeg(), *t=(char*)NULLP;
slen_t len=img->getRlen()*img->getHt();
// fprintf(stderr, "len=%u\n", len);
register unsigned smallen, i;
switch (head[1]) {
case '1': /* PBM ASCII */
while (len>=70) {
smallen=70; t=tmp; while (smallen--!=0) *t++=(*p++==0)?'1':'0';
/* ^^^ note the swapping of '0' and '1' above */
*t++='\n'; out.vi_write(tmp, 71);
len-=70;
}
while (len--!=0) *t++=(*p++==0)?'0':'1';
/* Dat: xv requires a whitespace just before EOF */
*t++='\n'; out.vi_write(tmp, t-tmp);
break;
case '2': case '3': /* PGM ASCII, PPM ASCII */
while (len!=0) {
if (len>17) { smallen=17; len-=17; } else { smallen=len; len=0; }
t=tmp; while (smallen--!=0) {
if ((i=*(unsigned char*)p++)<10) *t++=i+'0';
else if (i<100) { *t++=i/10+'0'; *t++=i%10+'0'; }
else if (i<200) { *t++='1'; *t++=(i-100)/10+'0'; *t++=i%10+'0'; }
else { *t++='2'; *t++=(i-200)/10+'0'; *t++=i%10+'0'; }
*t++=' ';
}
/* Dat: xv requires a whitespace just before EOF */
t[-1]='\n'; out.vi_write(tmp, t-tmp/*-(len==0)*/);
}
break;
case '4': /* PBM RAWBITS */
/* Invert the image */
while (len--!=0) *p++^=-1;
p=img->getRowbeg();
len=img->getRlen()*img->getHt();
/* fall through */
default: /* PBM RAWBITS, PGM RAWBITS, PPM RAWBITS */
/* fwrite(p, 1, len, stdout); */
out.vi_write(p, len);
}
if (alphaChannel!=NULLP) {
/* OK: don't always output rawbits PBM file */
assert(alphaChannel->getBpc()==1);
assert(alphaChannel->getWd()==img->getWd());
assert(alphaChannel->getHt()==img->getHt());
/* write PBM RAWBITS subfile (alpha channel) */
if (or_->cache.TransferEncoding==or_->cache.TE_Binary) {
out << "P4 " << img->getWd() << ' ' << img->getHt() << '\n';
out.vi_write(alphaChannel->getRowbeg(), alphaChannel->getRlen()*alphaChannel->getHt());
/* ^^^ BUGFIX at Tue Sep 17 10:18:28 CEST 2002 */
} else {
out << "P1 " << img->getWd() << ' ' << img->getHt() << '\n';
alphaChannel->to8();
p=alphaChannel->getRowbeg(); len=alphaChannel->getRlen()*alphaChannel->getHt();
while (len>=70) {
smallen=70; t=tmp; while (smallen--!=0) *t++=(*p++!=0)?'1':'0';
/* ^^^ note the non-swapping of '0' and '1' above */
*t++='\n'; out.vi_write(tmp, 71);
len-=70;
}
while (len--!=0) *t++=(*p++!=0)?'0':'1';
/* Dat: xv requires a whitespace just before EOF */
*t++='\n'; out.vi_write(tmp, t-tmp);
}
delete alphaChannel;
}
return Rule::Applier::OK;
}
Rule::Applier out_pnm_applier = { "PNM", out_pnm_check_rule, out_pnm_work, 0 };
/* --- Sat Aug 10 22:18:33 CEST 2002 */
Rule::Applier::cons_t out_xwd_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_XWD
) return Rule::Applier::DONT_KNOW;
bool badp=false;
if (cache->SampleFormat!=Image::SF_Rgb8 && cache->SampleFormat!=Image::SF_Gray8 && cache->SampleFormat!=Image::SF_Indexed8) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /XWD must be /Rgb8, /Gray8 or /Indexed8" << (Error*)0;
return Rule::Applier::BAD;
}
if (cache->TransferEncoding!=cache->TE_Binary) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /XWD requires /Binary" << (Error*)0;
badp=true;
}
if (cache->Compression!=Rule::Cache::CO_None) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /XWD requires /Compression/None" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /XWD requires /Predictor 1" << (Error*)0;
badp=true;
}
if (badp) return Rule::Applier::BAD;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_xwd_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
static const unsigned
XWD_FILE_VERSION=7,
ZPixmap=2,
MSBFirst=1,
DirectColor=5,
PseudoColor=3,
//GrayScale=1,
StaticGray=0;
Image::sf_t sfo=or_->cache.SampleFormat;
if (out_xwd_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
or_->doSampleFormat(sf);
sfo=or_->cache.SampleFormat;
char head[101], *p=head;
Image::Sampled *img=sf->getImg();
slen_t bits_per_pixel, bitmap_pad, bytes_per_line;
unsigned ncolors;
memset(head, '\0', 101);
/*header_size*/ mf32(p,101);
/*file_version*/ (p+=4)[-1]=XWD_FILE_VERSION;
/*pixmap_format*/ (p+=4)[-1]=ZPixmap;
/*pixmap_depth*/ (p+=4)[-1]=sfo==Image::SF_Rgb8 ? 24 : 8;
/*pixmap_width */ mf32(p, img->getWd());
/*pixmap_height*/ mf32(p, img->getHt());
/*xoffset*/ p+=4;
/*byte_order*/ (p+=4)[-1]=MSBFirst;
/*bitmap_unit*/ (p+=4)[-1]=8; // sfo==Image::SF_Rgb8 ? 32 : 8;
/*bitmap_bit_order*/ (p+=4)[-1]=MSBFirst;
/*bitmap_pad*/ bitmap_pad=(p+=4)[-1]=8; // sfo==Image::SF_Rgb8 ? 32 : 8;
/* ^^^ force no padding at all */
/*bits_per_pixel*/ bits_per_pixel=(p+=4)[-1]=sfo==Image::SF_Rgb8 ? 24 : 8;
/*bytes_per_line*/ mf32(p, bytes_per_line=((bits_per_pixel*img->getWd()+bitmap_pad-1)&~(bitmap_pad-1))>>3);
/*visual_class*/ (p+=4)[-1]=sfo==Image::SF_Rgb8 ? DirectColor : sfo==Image::SF_Indexed8 ? PseudoColor : StaticGray;
/*red_mask */ (p+=4)[-3]=(char)(sfo==Image::SF_Rgb8 ? 255 : 0);
/*green_mask*/ (p+=4)[-2]=(char)(sfo==Image::SF_Rgb8 ? 255 : 0);
/*blue_mask */ (p+=4)[-1]=(char)(sfo==Image::SF_Rgb8 ? 255 : 0);
/*bits_per_rgb*/ (p+=4)[-1]=sfo==Image::SF_Rgb8 ? 24 : 8;
/*colormap_entries*/ (p+=4)[-2]=1; /*256*/
/*ncolors*/ mf32(p, ncolors=sfo==Image::SF_Rgb8 ? 0 : sfo==Image::SF_Indexed8 ? ((Image::Indexed*)img)->getNcols() : 256);
/*window_width */ mf32(p, img->getWd());
/*window_height*/ mf32(p, img->getHt());
assert(p+13==head+101);
/*window_x*/ /*0*/
/*window_y*/ /*0*/
/*window_bdrwidth*/ /*0*/
/*filename*/ /*""*/
out.vi_write(head, 101);
if (sfo!=Image::SF_Rgb8) {
char *pal=new char[ncolors*12], *pp=pal;
unsigned pixel;
if (sfo==Image::SF_Indexed8) {
char const* q=img->getHeadp();
for (pixel=0;pixel<ncolors;pixel++) {
*pp++=0; *pp++=0; *pp++=0; *pp++=pixel;
*pp++=*q; *pp++=*q++; /* red */
*pp++=*q; *pp++=*q++; /* green */
*pp++=*q; *pp++=*q++; /* blue */
*pp++=7;
*pp++=0;
}
} else { assert(sfo==Image::SF_Gray8);
for (pixel=0;pixel<ncolors;pixel++) {
*pp++=0; *pp++=0; *pp++=0; *pp++=pixel;
*pp++=pixel; *pp++=pixel; /* red */
*pp++=pixel; *pp++=pixel; /* green */
*pp++=pixel; *pp++=pixel; /* blue */
*pp++=7;
*pp++=0;
}
}
out.vi_write(pal, ncolors*12);
delete [] pal;
}
slen_t rlen=img->getRlen();
Image::Sampled::dimen_t htc=img->getHt();
char const* rp=img->getRowbeg();
unsigned scanline_pad=bytes_per_line-rlen;
// assert(*rp=='\xff');
if (scanline_pad!=0) {
assert(1<=scanline_pad && scanline_pad<=3);
while (htc--!=0) {
out.vi_write(rp, rlen);
rp+=rlen;
if (scanline_pad!=0) out.vi_write("\0\0", scanline_pad);
}
} else out.vi_write(rp, rlen*htc);
return Rule::Applier::OK;
}
Rule::Applier out_xwd_applier = { "XWD", out_xwd_check_rule, out_xwd_work, 0 };
Rule::Applier out_x11_applier = { "X11", 0/*out_x11_check_rule*/, 0/*out_x11_work*/, 0 };
/* --- Sat Apr 20 11:49:56 CEST 2002 */
/** Baseline (lossy) JPEG */
Rule::Applier::cons_t out_jpeg_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_JPEG
|| cache->Compression==Rule::Cache::CO_JAI
) return Rule::Applier::DONT_KNOW;
bool badp=false;
if (cache->SampleFormat!=Image::SF_Rgb8 && cache->SampleFormat!=Image::SF_Gray8) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /DCTEncode requires /Rgb8 or /Gray8" << (Error*)0;
badp=true;
}
if (cache->TransferEncoding!=cache->TE_Binary) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /JPEG requires /Binary" << (Error*)0;
badp=true;
}
if (cache->Compression!=Rule::Cache::CO_None && cache->Compression!=Rule::Cache::CO_DCT && cache->Compression!=Rule::Cache::CO_IJG) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /JPEG requires /DCT or /IJG" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /JPEG requires /Predictor 1" << (Error*)0;
badp=true;
}
if (badp) return Rule::Applier::BAD;
or_->cache.WarningOK=true; /* ?? */
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_jpeg_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
Rule::Cache *cache=&or_->cache;
// assert(0);
if (out_jpeg_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
or_->doSampleFormat(sf);
// GenBuffer::Writable *tp=&out; /* always binary */
GenBuffer::Writable *cp=&out;
/* SUXX: cjpeg(1) won't create a color JPEG for a grayscale image */
if (cache->Compression==Rule::Cache::CO_DCT) {
SimBuffer::B other_parameters;
or_->cacheHints.DCT->dump(other_parameters, 0, false);
cp=PSEncoder::newDCTEncode(out, or_->cacheHints.EncoderColumns, or_->cacheHints.EncoderRows, or_->cacheHints.EncoderColors, or_->cacheHints.ColorTransform, other_parameters);
} else {
assert(cache->Compression==Rule::Cache::CO_None || cache->Compression==Rule::Cache::CO_IJG);
cp=PSEncoder::newDCTIJGEncode(out, or_->cacheHints.EncoderColumns, or_->cacheHints.EncoderRows, or_->cacheHints.EncoderColors, or_->cacheHints.Quality);
}
Rule::writePalData(out, *cp, sf);
delete cp;
return Rule::Applier::OK;
}
Rule::Applier out_jpeg_applier = { "JPEG", out_jpeg_check_rule, out_jpeg_work, 0 };
/* --- Wed Apr 17 13:32:46 CEST 2002 */
Rule::Applier::cons_t out_jpegjai_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_JPEG
|| cache->Compression!=Rule::Cache::CO_JAI
) return Rule::Applier::DONT_KNOW;
bool badp=false;
if (cache->SampleFormat!=Image::SF_Asis) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/JPEG /Compression/JAI requires /SampleFormat/Asis" << (Error*)0;
badp=true;
}
if (cache->TransferEncoding!=cache->TE_Binary) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/JPEG /Compression/JAI requires /TransferEncoding/Binary" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/JPEG /Compression/JAI requires /Predictor 1" << (Error*)0;
badp=true;
}
if (badp) return Rule::Applier::BAD;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_jpegjai_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
/* This is the second simplest Applier I've ever written. */
if (out_jpegjai_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
Image::Sampled *img=sf->getImg();
// out.vi_write(img->getHeadp(), img->end_()-img->getHeadp());
/* ^^^ end_() BUGFIX by
[email protected] at Sun Jun 2 22:24:32 CEST 2002 */
/* ^^^ end_() contains 8 extra bytes */
out.vi_write(img->getHeadp(), img->getRowbeg()-img->getHeadp());
return Rule::Applier::OK;
}
Rule::Applier out_jpegjai_applier = { "JPEG-JAI", out_jpegjai_check_rule, out_jpegjai_work, 0 };
/* --- Sun Jun 2 22:25:16 CEST 2002 */
class TIFFPrinter {
GenBuffer::Writable& out;
/** true iff little endian == LSB first */
bool le;
SimBuffer::B s, dir;
public:
TIFFPrinter(GenBuffer::Writable &out_, bool le_);
inline SimBuffer::B& getS() { return s; }
inline SimBuffer::B const& getDir() const { return dir; }
void aSHORT(SimBuffer::B &s, unsigned count, unsigned short const*val);
void aLONG (SimBuffer::B &s, unsigned count, slen_t const*val);
void dirSHORT(unsigned short tag, slen_t count, unsigned short const*val);
void dirLONG(unsigned short tag, slen_t count, slen_t const*val);
void dirSL(unsigned short tag, slen_t val);
void dirLONG(unsigned short tag, slen_t val);
void dirRATIONAL(unsigned short tag, slen_t count, slen_t const*val);
void dirUNDEFINED(unsigned short tag, slen_t count, char const*val);
void dirUNDEFINED(unsigned short tag, slen_t count, char const*val, slen_t count2, char const*val2);
void dirClose();
BEGIN_STATIC_ENUM1(unsigned short)
EXTRASAMPLE_ASSOCALPHA=1, /* !associated alpha data */
EXTRASAMPLE_UNASSALPHA=2 /* !unassociated alpha data */
END_STATIC_ENUM()
BEGIN_STATIC_ENUM1(unsigned short)
COMPRESSION_NONE =1, /* dump mode */
COMPRESSION_CCITTRLE =2, /* CCITT modified Huffman RLE; unused in sam2p */
COMPRESSION_CCITTFAX3=3, /* CCITT Group 3 fax encoding */
COMPRESSION_CCITTFAX4=4, /* CCITT Group 4 fax encoding */
COMPRESSION_LZW =5, /* Lempel-Ziv & Welch */
COMPRESSION_OJPEG =6, /* !6.0 JPEG; obsolete, unused in sam2p */
COMPRESSION_JPEG =7, /* %JPEG DCT compression */
COMPRESSION_CCITTRLEW=32771,/* #1 w/ word alignment; unused in sam2p */
COMPRESSION_PACKBITS =32773,/* Macintosh RLE */
COMPRESSION_DEFLATE =32946 /* Deflate compression */
END_STATIC_ENUM()
BEGIN_STATIC_ENUM1(unsigned short)
PHOTOMETRIC_MINISBLACK=1,
PHOTOMETRIC_RGB=2,
PHOTOMETRIC_PALETTE=3,
PHOTOMETRIC_MASK=4,
PHOTOMETRIC_SEPARATED=5, /* possibly CMYK */
PHOTOMETRIC_YCBCR=6
END_STATIC_ENUM()
BEGIN_STATIC_ENUM1(unsigned short)
GROUP3OPT_2DENCODING=0x1,
GROUP3OPT_UNCOMPRESSED=0x2,
GROUP3OPT_FILLBITS=0x4,
GROUP4OPT_UNCOMPRESSED=0x2
END_STATIC_ENUM()
BEGIN_STATIC_ENUM1(unsigned short)
ImageWidth=256,
ImageLength=257,
BitsPerSample=258,
Compression=259,
Photometric=262,
FillOrder=266,
StripOffsets=273,
SamplesPerPixel=277,
RowsPerStrip=278,
StripByteCounts=279,
XResolution=282,
YResolution=283,
PlanarConfig=284,
Group3Options=292,
Group4Options=293,
ResolutionUnit=296,
Predictor=317,
ColorMap=320,
InkSet=332,
ExtraSamples=338,
JPEGTables=347,
YCbCrSubsampling=530,
ReferenceBlackWhite=532
END_STATIC_ENUM()
};
TIFFPrinter::TIFFPrinter(GenBuffer::Writable &out_, bool le_) :out(out_), le(le_) {
/* Directory now at unknown offset (will be set by .dirClose()) */
s.vi_write(le_ ? "II*\0\0\0\0\0" : "MM\0*\0\0\0\0", 8);
dir.vi_write("\0",2); /* initial number of directory entries */
}
void TIFFPrinter::aSHORT(SimBuffer::B &s, unsigned count, unsigned short const*val) {
if (le) while (count--!=0) { s << (char)(*val) << (char)(*val>>8); val++; }
else while (count--!=0) { s << (char)(*val>>8) << (char)(*val); val++; }
}
void TIFFPrinter::aLONG (SimBuffer::B &s, unsigned count, slen_t const*val) {
/* Imp: is vi_write(..., 4); faster ? */
if (le) while (count--!=0) { s << (char)(*val) << (char)(*val>>8) << (char)(*val>>16) << (char)(*val>>24); val++; }
else while (count--!=0) { s << (char)(*val>>24) << (char)(*val>>16) << (char)(*val>>8) << (char)(*val); val++; }
}
void TIFFPrinter::dirSHORT(unsigned short const tag, slen_t const count, unsigned short const*val) {
slen_t offs;
aSHORT(dir, 1U, &tag);
dir.vi_write(&("\0\3"[le?1:0]), 2);
aLONG(dir, 1U, &count);
switch (count) {
case 0: dir.vi_write("\0\0\0", 4); break;
case 1: aSHORT(dir, 1, val); dir.vi_write("\0", 2); break;
case 2: aSHORT(dir, 2, val); break;
default:offs=s.getLength(); aSHORT(s, count, val); aLONG(dir, 1, &offs);
}
}
void TIFFPrinter::dirLONG(unsigned short const tag, slen_t const count, slen_t const*val) {
slen_t offs;
aSHORT(dir, 1U, &tag);
dir.vi_write(&("\0\4"[le?1:0]), 2);
aLONG(dir, 1U, &count);
switch (count) {
case 0: dir.vi_write("\0\0\0", 4); break;
case 1: aLONG(dir, 1, val); break;
default:offs=s.getLength(); aLONG(s, count, val); aLONG(dir, 1, &offs);
}
}
void TIFFPrinter::dirRATIONAL(unsigned short tag, slen_t count, slen_t const*val) {
slen_t offs;
aSHORT(dir, 1U, &tag);
dir.vi_write(&("\0\5"[le?1:0]), 2);
aLONG(dir, 1U, &count);
switch (count) {
case 0: dir.vi_write("\0\0\0", 4); break;
default:offs=s.getLength(); aLONG(s, count*2, val); aLONG(dir, 1, &offs);
}
}
void TIFFPrinter::dirUNDEFINED(unsigned short tag, slen_t count, char const*val) {
slen_t offs;
aSHORT(dir, 1U, &tag);
dir.vi_write(&("\0\7"[le?1:0]), 2);
aLONG(dir, 1U, &count);
if (count<=4) {
dir.vi_write(val, count);
if (count!=4) dir.vi_write("\0\0\0", 4-count);
} else {
offs=s.getLength(); s.vi_write(val, count); aLONG(dir, 1, &offs);
}
}
void TIFFPrinter::dirUNDEFINED(unsigned short tag, slen_t count, char const*val, slen_t count2, char const*val2) {
slen_t offs, countx=count+count2;
aSHORT(dir, 1U, &tag);
dir.vi_write(&("\0\7"[le?1:0]), 2);
aLONG(dir, 1U, &countx);
if (countx<=4) {
dir.vi_write(val, count);
dir.vi_write(val2, count2);
if (countx!=4) dir.vi_write("\0\0\0", 4-countx);
} else {
offs=s.getLength();
s.vi_write(val, count);
s.vi_write(val2, count2);
aLONG(dir, 1, &offs);
}
}
void TIFFPrinter::dirSL(unsigned short tag, slen_t val) {
unsigned short sh;
if (val<1<<16U) { sh=val; dirSHORT(tag, 1, &sh); }
else dirLONG(tag, 1, &val);
}
void TIFFPrinter::dirLONG(unsigned short tag, slen_t val) {
dirLONG(tag, 1, &val);
}
void TIFFPrinter::dirClose() {
char *ss=s.begin_();
slen_t len=s.getLength();
if (le) { ss[4]=len; ss[5]=len>>8; ss[6]=len>>16; ss[7]=len>>24; }
else { ss[4]=len>>24; ss[5]=len>>16; ss[6]=len>>8; ss[7]=len; }
/* ^^^ dir offset: end of file */
out.vi_write(ss, len);
assert(dir.getLength()%12==2);
unsigned dirc=(dir.getLength()-2)/12;
ss=dir.begin_();
if (le) { ss[0]=dirc; ss[1]=dirc>>8; }
else { ss[0]=dirc>>8; ss[1]=dirc; }
out.vi_write(dir(), dir.getLength());
out.vi_write("", 0);
}
/* tiffjai --- Sun Jun 2 22:25:16 CEST 2002 */
Rule::Applier::cons_t out_tiffjai_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_TIFF
|| cache->Compression!=Rule::Cache::CO_JAI
) return Rule::Applier::DONT_KNOW;
bool badp=false;
if (cache->SampleFormat!=Image::SF_Asis) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/TIFF /Compression/JAI requires /SampleFormat/Asis" << (Error*)0;
badp=true;
}
if (!cache->isBinSB()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/TIFF /TransferEncoding/Binary|/?SBFirst" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/TIFF /Compression/JAI requires /Prediror 1" << (Error*)0;
badp=true;
}
if (badp) return Rule::Applier::BAD;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_tiffjai_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
if (out_tiffjai_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
Image::Sampled *img=sf->getImg();
unsigned char cs=img->getCs(); /* color space */
if (cs!=Image::Sampled::CS_GRAYSCALE && cs!=Image::Sampled::CS_RGB && cs!=Image::Sampled::CS_YCbCr && cs!=Image::Sampled::CS_CMYK) {
/* Dat: CS_YCCK is supported by JPEG, but unsupported by TIFF-JPEG */
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/TIFF /Compression/JAI doesn't support this color space" << (Error*)0;
return Rule::Applier::BAD;
}
/* Imp: test all CMYK TIFF tags */
/* Imp: test with RGB and CMYK images */
TIFFPrinter tp(out, or_->cache.TransferEncoding==or_->cache.TE_LSBfirst);
/* ^^^ Dat: Binary defaults to MSBfirst, beacuse QuarkXPress 3 can read only
* MSBfirst TIFF files.
*/
char const *databeg=img->getHeadp()+img->getXoffs();
assert(databeg[-2]=='\xFF' && databeg[-1]=='\xC0');
slen_t datalen=img->getRowbeg()-databeg;
// printf("LEN=%u gl=%u datalen=%u\n", img->end_()-img->begin_(), img->getLength(), datalen);
tp.getS().vi_write("\xFF\xD8\xFF\xC0", 4); /* fake-SOI SOF0 */
tp.getS().vi_write(databeg, datalen); /* DataPart */
// tp.getS().vi_write("\xFF\xD9", 2); /* fake-EOI */
/* ^^^ there must be an EOI already */
unsigned phot=0; bool inks=false, refe=false;
switch (cs) {
case Image::Sampled::CS_GRAYSCALE: phot=tp.PHOTOMETRIC_MINISBLACK; break;
case Image::Sampled::CS_RGB: phot=tp.PHOTOMETRIC_RGB; break;
case Image::Sampled::CS_YCbCr: phot=tp.PHOTOMETRIC_YCBCR; refe=true; break; /* preferred to RGB */
case Image::Sampled::CS_CMYK: phot=tp.PHOTOMETRIC_SEPARATED; inks=true; break; /* preferred to RGB */
default: Error::sev(Error::EERROR) << "TIFF6-JAI: color space " << (unsigned)cs << " not supported in TIFF-JPEG" << (Error*)0;
}
/* Dat: TIFF tags must appear in increasing numerical order */
tp.dirSL(tp.ImageWidth, img->getWd());
tp.dirSL(tp.ImageLength, img->getHt());
unsigned short eights[]={8,8,8,8};
tp.dirSHORT(tp.BitsPerSample, img->cs2cpp[cs], eights);
tp.dirSL(tp.Compression, tp.COMPRESSION_JPEG); /* SHORT */
tp.dirSL(tp.Photometric, phot); /* SHORT */
tp.dirLONG(tp.StripOffsets, 8);
tp.dirSL(tp.SamplesPerPixel, img->cs2cpp[cs]); /* SHORT */
tp.dirSL(tp.RowsPerStrip, img->getHt());
tp.dirLONG(tp.StripByteCounts, datalen+6);
slen_t rats[]={1,1, 0,1, 255,1, 128,1, 255,1, 128,1, 255,1};
tp.dirRATIONAL(tp.XResolution, 1, rats);
tp.dirRATIONAL(tp.YResolution, 1, rats);
tp.dirSL(tp.PlanarConfig, 1); /* SHORT, PLANARCONFIG_CONTIG */
tp.dirSL(tp.ResolutionUnit, 1); /* SHORT */
if (inks) tp.dirSL(tp.InkSet, 1); /* SHORT */
tp.dirUNDEFINED(tp.JPEGTables, databeg-img->getHeadp()-2, img->getHeadp(), 2, "\xFF\xD9");
const unsigned char hvs=img->end_()[-1];
if (hvs!=0x22 && cs==Image::Sampled::CS_YCbCr) {
// printf("hvs=0x%02X\n", hvs);
const unsigned short horiz_vert[2]={ (unsigned short)((hvs+0U)>>4),
(unsigned short)(hvs&15U) };
tp.dirSHORT(tp.YCbCrSubsampling, 2, horiz_vert);
}
if (refe) tp.dirRATIONAL(tp.ReferenceBlackWhite, 6, rats+2);
tp.dirClose();
#if 0
{ Files::FILEW f(fopen("t.hea","wb"));
f.vi_write(img->getHeadp(), databeg-img->getHeadp()-2); /* HeadPart */
f.vi_write("\xFF\xD9", 2);
f.close();
}
{ Files::FILEW f(fopen("t.dat","wb"));
f.vi_write("\xFF\xD8\xFF\xC0", 4); /* fake-SOI SOF0 */
f.vi_write(databeg, datalen); /* DataPart */
f.close();
}
#endif
// out.vi_write(img->getHeadp(), img->getRowbeg()-img->getHeadp()); /* HeadPart */
// out.vi_write(img->getRowbeg(), img->end_()-img->getRowbeg()); /* DataPart */
return Rule::Applier::OK;
/* SUXX: see compatibility notes in FAQ (README) */
}
Rule::Applier out_tiffjai_applier = { "TIFF6-JAI", out_tiffjai_check_rule, out_tiffjai_work, 0 };
/* tiff --- Tue Jun 4 16:23:48 CEST 2002 */
/** by
[email protected] at Tue Jun 11 16:56:53 CEST 2002
* Reads a JPEG Baseline (SOF0) stream on input, and drops everything before
* the first SOF0 marker. Then writes an SOI marker, the SOF0 marker, and
* everything after the SOI marker. The last two bytes read must be an
* EOI marker (/\xFF\xD9/). The first thing to be read must be a SOI marker
* (/\xFF+\xD8/). The getJPEGTables() method can be used to retrieve the
* data read before the SOF0 marker. An SOS marker must be read after SOF0.
* Also retains some information (hvsamples[0]) from the SOF0 marker.
* Suitable for reading the output of cjpeg and Ghostscript /FlateEncode.
* This encoder would be much harder if we were not allowed to read the
* JPEG stream into memory.
*/
class JPEGSOF0Encode: public PSEncoder {
public:
/** @param maxcpl_: maximum # hex digits per line, should be even */
JPEGSOF0Encode(GenBuffer::Writable &out_);
virtual void vi_write(char const*buf, slen_t len);
SimBuffer::Flat const& getJPEGTables() const { return buf; }
unsigned char getColorSpace() const { return gi.colorspace; }
unsigned char getHVS() const { return gi.hvs; }
protected:
GenBuffer::Writable &out;
SimBuffer::B buf;
struct jai_gfxinfo gi;
};
JPEGSOF0Encode::JPEGSOF0Encode(GenBuffer::Writable &out_): out(out_) {}
void JPEGSOF0Encode::vi_write(char const*bufr, slen_t len) {
if (len==0) {
{ Filter::FlatD flatd(buf(), buf.getLength());
jai_parse_jpeg(&gi, &flatd);
}
slen_t len=buf.getLength();
if (gi.bad==0 && (len<4 || buf[len-2]!='\xFF' || buf[len-1]!='\xD9')) gi.bad=10;
if (gi.bad!=0)
Error::sev(Error::EERROR) << "JPEGS0F0: invalid JPEG stream: " << jai_errors[gi.bad] << (Error*)0;
out.vi_write("\xFF\xD8", 2); /* extra SOI */
out.vi_write(buf()+gi.SOF_offs-2, len-gi.SOF_offs+2); /* SOF0 and followers */
out.vi_write(0,0);
assert(buf[gi.SOF_offs-2]=='\xFF'); /* BUGFIX at 2002.12.02 */
buf[gi.SOF_offs-1]='\xD9'; /* extra EOI */
buf.keepLeft(gi.SOF_offs); /* keep only headers */
} else buf.vi_write(bufr, len);
}
Rule::Applier::cons_t out_tiff_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_TIFF
|| cache->Compression==Rule::Cache::CO_JAI
) return Rule::Applier::DONT_KNOW;
bool badp=false;
/* Dat: acroread TIFF predictor2 OK. (examples/fishg_lzw2_pdf.job) */
/* Dat: /ZIP with /Predictor OK */
if (cache->Predictor!=cache->PR_None) {
if (cache->Predictor!=cache->PR_TIFF2) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/TIFF requires /Predictor 1|2" << (Error*)0;
badp=true;
}
if (cache->Compression!=Rule::Cache::CO_ZIP && cache->Compression!=Rule::Cache::CO_LZW) {
Error::sev(Error::WARNING_DEFER) << "check_rule: real /Predictor requires /ZIP or /LZW" << (Error*)0;
badp=true;
}
}
#if !HAVE_LZW
if (cache->Compression==Rule::Cache::CO_LZW) {
Error::sev(Error::WARNING_DEFER) << "check_rule: `configure --enable-lzw' for /Compression/LZW with /FileFormat/TIFF" << (Error*)0;
badp=true;
}
#endif
if (cache->Compression==Rule::Cache::CO_Fax && !cache->isOneBit()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/TIFF /Compression/Fax requires a 1-bit /SampleFormat" << (Error*)0;
badp=true;
}
if (!cache->isGray() && !cache->isTransparentM() && !cache->isIndexed() && !cache->isRGB()) {
/* Dat: unsupported SampleFormats: /Opaque, /Transparent and /Asis. */
Error::sev(Error::WARNING_DEFER) << "check_rule: unsupported /SampleFormat for /FileFormat/TIFF" << (Error*)0;
badp=true;
}
if (!cache->isBinSB()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /FileFormat/TIFF requires /TransferEncoding/Binary|/?SBFirst" << (Error*)0;
badp=true;
}
if (cache->isDCTE() && cache->SampleFormat!=Image::SF_Rgb8 && cache->SampleFormat!=Image::SF_Gray8 && cache->SampleFormat!=Image::SF_Indexed8) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /DCTEncode requires /Rgb8 or /Gray8 (or /Indexed8)" << (Error*)0;
badp=true;
}
if (!badp && cache->isTransparentM()) {
cache->origSampleFormat=cache->SampleFormat;
cache->SampleFormat=Image::SF_Transparent8;
}
return (Rule::Applier::cons_t)(badp ? 0+Rule::Applier::BAD : 0+Rule::Applier::OK);
/* ^^^ Dat: 0+: pacify g++-3.1 */
}
Rule::Applier::cons_t out_tiff_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
Rule::Cache *cache=&or_->cache;
Image::sf_t origSampleFormat=cache->origSampleFormat;
if (out_tiff_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
assert(!sf->hasTranspp() || cache->isTransparentM());
/** Alpha channel or null. */
Image::Indexed *alpha=(Image::Indexed*)NULLP;
if (cache->isTransparentM()) {
// fprintf(stderr,"sf=%u osf=%u\n", cache->SampleFormat, origSampleFormat);
alpha=PTS_dynamic_cast(Image::Indexed*,sf->getImg())->calcAlpha();
PTS_dynamic_cast(Image::Indexed*,sf->getImg())->getClearTransp();
assert((alpha!=NULLP) == (sf->hasTranspp()==true));
sf->clearTransp();
static Image::sf_t const graytab[9]={Image::SF_None,Image::SF_Gray1,Image::SF_Gray2,Image::SF_None,Image::SF_Gray4,Image::SF_None,Image::SF_None,Image::SF_None,Image::SF_Gray8};
static Image::sf_t const rgbtab[9]={Image::SF_None,Image::SF_Rgb1,Image::SF_Rgb2,Image::SF_None,Image::SF_Rgb4,Image::SF_None,Image::SF_None,Image::SF_None,Image::SF_Rgb8};
static Image::sf_t const indexedtab[9]={Image::SF_None,Image::SF_Indexed1,Image::SF_Indexed2,Image::SF_None,Image::SF_Indexed4,Image::SF_None,Image::SF_None,Image::SF_None,Image::SF_Indexed8};
unsigned char minbpc=sf->minRGBBpcc();
// fprintf(stderr,"minbpc=%u\n",minbpc);
if (minbpc<8 && origSampleFormat==Image::SF_Transparent8) minbpc=8;
else if (minbpc<4 && origSampleFormat==Image::SF_Transparent4) minbpc=4;
else if (minbpc<2 && origSampleFormat==Image::SF_Transparent2) minbpc=2;
// fprintf(stderr,"minbpc=%u\n",minbpc);
// cacheHints.EncoderBPL=(slen_t)img->getWd()*img->getCpp()*img->getBpc();
/* ^^^ Dat: doSampleFormat will do it correctly */
// assert(saf!=Image::SF_max);
cache->SampleFormat=(alpha==NULLP ? indexedtab : sf->minGrayBpcc()==0 ? rgbtab : graytab)[minbpc];
}
or_->doSampleFormat(sf); /* No separations */
Image::Sampled *img=sf->getImg(); /* call this _after_ doSampleFormat()! */
TIFFPrinter tp(out, or_->cache.TransferEncoding==or_->cache.TE_LSBfirst);
/* ^^^ Dat: Binary defaults to MSBfirst, beacuse QuarkXPress 3 can read only
* MSBfirst TIFF files.
*/
unsigned phot= cache->isRGB() ? 0+tp.PHOTOMETRIC_RGB
// : cache->SampleFormat==Image::SF_Mask ? tp.PHOTOMETRIC_MASK
: cache->isGray() ? 0+tp.PHOTOMETRIC_MINISBLACK
// : cache->isIndexed() ? tp.PHOTOMETRIC_PALETTE
: 0+tp.PHOTOMETRIC_PALETTE /* /Indexed*, /Mask, /Transparent+ */;
unsigned compr=tp.COMPRESSION_NONE;
Filter::VerbatimCountE vc(tp.getS());
GenBuffer::Writable *cp=&vc;
JPEGSOF0Encode *jp=(JPEGSOF0Encode*)NULLP;
// Dat: g++-3.4: Rule::Cache::CO_None: `cache cannot appear in a constant expression'
switch (cache->Compression) {
case Rule::Cache::CO_None: break;
case Rule::Cache::CO_ZIP: compr=tp.COMPRESSION_DEFLATE; cp=PSEncoder::newFlateEncode(*cp, or_->cacheHints.Effort); break;
case Rule::Cache::CO_LZW: compr=tp.COMPRESSION_LZW; cp=PSEncoder::newLZWEncode(*cp); break;
/* vvv Dat: RunLengthEncode EOD (char 128) is OK and ignored in TIFF PackBits streams */
case Rule::Cache::CO_RLE: compr=tp.COMPRESSION_PACKBITS; cp=PSEncoder::newRunLengthEncode(*cp, or_->cacheHints.RecordSize); break;
case Rule::Cache::CO_Fax:
if (or_->cacheHints.K<0) compr=tp.COMPRESSION_CCITTFAX4;
else if (or_->cacheHints.K==0) compr=tp.COMPRESSION_CCITTFAX3;
else { compr=tp.COMPRESSION_CCITTFAX3; or_->cacheHints.K=img->getHt(); }
cp=PSEncoder::newCCITTFaxEncode(*cp, or_->cacheHints.K, or_->cacheHints.EncoderBPL, /*EndOfLine:*/ compr==tp.COMPRESSION_CCITTFAX3, /*BlackIs1: !*/ true);
/* Dat: libtiff enforces EndOfLine==(compr==tp.COMPRESSION_CCITTFAX3) */
break;
case Rule::Cache::CO_IJG:
compr=tp.COMPRESSION_JPEG;
jp=new JPEGSOF0Encode(vc);
cp=PSEncoder::newDCTIJGEncode(*jp, or_->cacheHints.EncoderColumns, or_->cacheHints.EncoderRows, or_->cacheHints.EncoderColors, or_->cacheHints.Quality);
break;
case Rule::Cache::CO_DCT: compr=tp.COMPRESSION_JPEG; {
SimBuffer::B other_parameters;
or_->cacheHints.DCT->dump(other_parameters, 0, false);
jp=new JPEGSOF0Encode(vc);
cp=PSEncoder::newDCTEncode(*jp, or_->cacheHints.EncoderColumns, or_->cacheHints.EncoderRows, or_->cacheHints.EncoderColors, or_->cacheHints.ColorTransform, other_parameters);
break; }
default: assert(0);
}
GenBuffer::Writable *pp=cp;
if (cache->hasPredictor()) pp=PSEncoder::newPredictor(*cp, cache->Predictor, or_->cacheHints.PredictorBPC, or_->cacheHints.PredictorColumns, or_->cacheHints.PredictorColors);
slen_t rlenht=img->getRlen()*img->getHt();
Image::Sampled::dimen_t wd=img->getWd();
if (rlenht!=0) {
/* Dat: TIFF inserts extra sample bits... */
/* TIFF images with transparency are really inefficient. That's because
* libtiff doesn't read or write an indexed image with transparency:
* Sorry, can not handle contiguous data with PhotometricInterpretation=3,
* and Samples/pixel=2.
* So Indexed images are converted to RGB* or Gray* first, and then the
* alpha channel is added. The result is a big and inefficiently
* compressible TIFF file.
*/
if (alpha!=NULLP) {
unsigned char bpc=img->getBpc();
slen_t rlen=img->getRlen();
slen_t writelen;
char *buf=(char*)NULLP;
char const*psave=img->getRowbeg(), *p, *ppend=psave+rlenht;
char const*r=alpha->getRowbeg();
#ifndef NDEBUG
char const*rend;
Image::Sampled::dimen_t rlena=(wd+7)>>3;
#endif
char *t;
register unsigned u;
#ifndef NDEBUG
assert(rlena==alpha->getRlen());
#endif
// printf("SF=%u\n", cache->SampleFormat);
if (cache->isGray()) {
/* works at Mon Dec 9 01:25:59 CET 2002 */
static unsigned char const szor1[16]={2*0,2*1,2*4,2*5,2*16,2*17,2*20,2*21,2*64,2*65,2*68,2*69,2*80,2*81,2*84,2*85};
static unsigned char const szor2[16]={85,84,81,80,69,68,65,64,21,20,17,16,5,4,1,0};
// static unsigned char const szor2[8]={0,1,16,17,64,65,80,81};
buf=new char[(writelen=((slen_t)wd*bpc+3)>>2)+24];
#ifndef NDEBUG
assert(rlena*8>=rlen);
#endif
while (psave!=ppend) {
t=buf;
#ifndef NDEBUG
rend=r+rlena;
#endif
p=psave; psave+=rlen;
assert(psave<=ppend);
u=(1<<16);
switch (bpc) {
case 8:
while (p!=psave) {
/* Dat: works at Tue Sep 10 22:45:52 CEST 2002 */
if (0!=(u&(1<<16))) u=*(unsigned char const*)r++|(1<<8);
*t++=*p++; *t++=-(0==(u&128));
u<<=1;
} break;
case 4:
while (p!=psave) {
if (0!=(u&(1<<16))) u=*(unsigned char const*)r++|(1<<8);
*t++=(p[0]&0xF0)|(0==(u&128)?15:0);
*t++=((p[0]&0xF)<<4)|(0==(u&64)?15:0);
p++; u<<=2;
} break;
case 2:
while (p!=psave) {
if (0!=(u&(1<<16))) u=*(unsigned char const*)r++|(1<<8);
*t++=(p[0]&0xC0 )|((u&128)!=0?0:48)|((p[0]&0x30)>>2)|((u&64)!=0?0:3);
*t++=((p[0]&0xC)<<4)|((u& 32)!=0?0:48)|((p[0]&0x3 )<<2)|((u&16)!=0?0:3);
p++; u<<=4;
} break;
case 1:
while (p!=psave) {
u=*(unsigned char const*)r++;
*t++=szor1[*(unsigned char const*)p>>4]|szor2[u>>4];
*t++=szor1[*(unsigned char const*)p&15]|szor2[u&15];
p++;
} break;
} /* SWITCH Gray bpc */
// assert(p-psave+0U==rlena*8U);
assert(p>=psave);
pp->vi_write(buf, writelen);
}
} else {
/* works at Sun Dec 8 23:30:17 CET 2002 */
assert(cache->isRGB());
buf=new char[(writelen=((slen_t)wd*bpc+1)>>1)+24];
while (psave!=ppend) {
t=buf;
p=psave; psave+=rlen;
assert(psave<=ppend);
u=(1<<16);
#ifndef NDEBUG
rend=r+rlena; /* superfluous */
#endif
switch (bpc) {
case 8:
while (p<psave) {
if (0!=(u&(1<<16))) u=*(unsigned char const*)r++|(1<<8);
*t++=*p++;
*t++=*p++;
*t++=*p++;
*t++=-(0==(u&128));
u<<=1;
} break;
case 4:
while (p<psave) {
if (0!=(u&(1<<16))) u=*(unsigned char const*)r++|(1<<8);
*t++=p[0]; /* R0 and G0 */
*t++=(p[1]&0xF0)|((u&128)!=0?0:15); /* B0 and A0 */
*t++=(p[1]<<4)|((p[2]>>4)&15); /* R1 and G1 */
*t++=(p[2]<<4)|((u&64)!=0?0:15); /* B1 and A1 */
p+=3; u<<=2;
} break;
case 2:
while (p<psave) {
if (0!=(u&(1<<16))) u=*(unsigned char const*)r++|(1<<8);
/* Dat: p[0]==R0G0B0R1 p[1]==G1B1R2G2 p[2]==B2R3G3B3 */
*t++=(p[0]&0xFC)|((u&128)!=0?0:3); /* R0G0B0 and A0 */
*t++=(p[0]<<6)|((p[1]>>2)&0x3C)|((u&64)!=0?0:3); /* R1 G1B1 A1 */
*t++=(p[1]<<4)|((p[2]>>4)&0x0C)|((u&32)!=0?0:3); /* R2G1 B1 A2 */
*t++=(p[2]<<2)|((u&16)!=0?0:3); /* R3G3B3 and A3 */
p+=3; u<<=4;
} break;
default: // case 1:
while (p<psave) {
u=*(unsigned char const*)r++;
/* Dat: p[0]==RGBRGBRG p[1]==BRGBRGBR p[2]==GBRGBRGB */
*t++=(p[0]&0xE0)|((u&128)!=0?0:16)
|((p[0]>>1)&0xE)|((u&64)!=0?0:1);
*t++=(p[0]<<6)|((p[1]>>2)&0x80)|((u&32)!=0?0:16)
|((p[1]>>3)&0xE)|((u&16)!=0?0:1);
*t++=((p[1]<<4)&0xE0)|((u&8)!=0?0:16)
|((p[1]<<3)&0x7)|((p[2]>>5)&0x06)|((u&4)!=0?0:1);
*t++=((p[2]<<2)&0xE0)|((u&2)!=0?0:16)
|((p[2]<<1)&0x0E)|((u&1)!=0?0:1);
p+=3;
}
break;
} /* SWITCH RGB bpc */
#ifndef NDEBUG
assert(r==rend); // r=rend;
#endif
pp->vi_write(buf, writelen);
}
}
delete [] buf;
} else pp->vi_write(img->getRowbeg(), rlenht);
}
pp->vi_write(0,0); /* flush all */
if (pp!=cp) delete pp;
if (cp!=&vc) delete cp;
bool refe=false;
if (compr==tp.COMPRESSION_JPEG) switch (jp->getColorSpace()) {
case Image::Sampled::CS_GRAYSCALE:
assert(phot==tp.PHOTOMETRIC_MINISBLACK || phot==tp.PHOTOMETRIC_PALETTE);
/* Dat: we don't change `phot' here, so JPEG compression can be applied
* to the palette indexes of indexed images :-)
*/
break;
case Image::Sampled::CS_RGB: phot=tp.PHOTOMETRIC_RGB; break;
case Image::Sampled::CS_YCbCr: phot=tp.PHOTOMETRIC_YCBCR; refe=true; break; /* preferred to RGB */
// case Image::Sampled::CS_CMYK: phot=tp.PHOTOMETRIC_SEPARATED; inks=true; break; /* preferred to RGB */
default: Error::sev(Error::EERROR) << "TIFF6: color space " << (unsigned)jp->getColorSpace() << " not supported in TIFF-JPEG" << (Error*)0;
}
/* Dat: TIFF tags must appear in increasing numerical order */
tp.dirSL(tp.ImageWidth, wd);
// tp.dirSL(tp.ImageWidth, img->getWd()/2);
tp.dirSL(tp.ImageLength, img->getHt());
unsigned short s4[]={img->getBpc(),img->getBpc(),img->getBpc(),img->getBpc()};
/* ^^^ these values may be different according to the TIFF6 spec, but
* libtiff can read TIFF files only with same BitsPerSamples.
*/
unsigned sppa=img->getCpp()+(alpha!=NULLP ? 1:0);
tp.dirSHORT(tp.BitsPerSample, sppa, s4);
tp.dirSL(tp.Compression, compr); /* SHORT */
tp.dirSL(tp.Photometric, phot); /* SHORT */
if (cache->Compression==Rule::Cache::CO_Fax) tp.dirSL(tp.FillOrder, 1); /* byte abcdefgh is (a<<7)+...+h */
tp.dirLONG(tp.StripOffsets, 8);
if (!cache->isIndexed())
tp.dirSL(tp.SamplesPerPixel, sppa); /* SHORT */
// tp.dirSL(tp.SamplesPerPixel, 2); /* SHORT */
tp.dirSL(tp.RowsPerStrip, img->getHt());
tp.dirLONG(tp.StripByteCounts, vc.getCount());
slen_t rats[]={1,1, 0,1, 255,1, 128,1, 255,1, 128,1, 255,1};
tp.dirRATIONAL(tp.XResolution, 1, rats);
tp.dirRATIONAL(tp.YResolution, 1, rats);
tp.dirSL(tp.PlanarConfig, 1); /* SHORT, PLANARCONFIG_CONTIG */
if (compr==tp.COMPRESSION_CCITTFAX3) tp.dirLONG(tp.Group3Options, (tp.GROUP3OPT_UNCOMPRESSED)|(or_->cacheHints.K!=0?tp.GROUP3OPT_2DENCODING:0));
if (compr==tp.COMPRESSION_CCITTFAX4) tp.dirLONG(tp.Group4Options, tp.GROUP4OPT_UNCOMPRESSED);
tp.dirSL(tp.ResolutionUnit, 1); /* SHORT */
if (cache->Predictor!=cache->PR_None) {
tp.dirSL(tp.Predictor, cache->Predictor); /* 1|2 */
if (img->getBpc()!=8)
Error::sev(Error::WARNING) << "TIFF6: libtiff supports /Predictor only with bpc=8 and bpc=16" << (Error*)0;
}
if (cache->isIndexed() || cache->isTransparentM()) {
unsigned colorlen=(1<<img->getBpc()), i=0;
unsigned short *r=new unsigned short[3*colorlen], *g=r+colorlen, *b=g+colorlen;
memset(r, '\0', 3*colorlen);
/** Image palette data */
char const *p=img->getHeadp(), *pend=img->getRowbeg();
assert((pend-p)%3==0);
assert(3*colorlen>=(unsigned)(pend-p));
while (p<pend) {
r[i ]=*(unsigned char const*)p++*257;
g[i ]=*(unsigned char const*)p++*257;
b[i++]=*(unsigned char const*)p++*257;
}
/* ^^^ In a TIFF palette (ColorMap) 0 is black, 65535 is lightest */
tp.dirSHORT(tp.ColorMap, colorlen*3, r);
delete [] r;
}
if (alpha!=NULLP) {
/* vvv Dat: GIMP 1.0 load fails with tp.EXTRASAMPLE_UNASSALPHA */
tp.dirSL(tp.ExtraSamples, tp.EXTRASAMPLE_ASSOCALPHA); /* SHORT */
delete alpha;
}
/* no tp.InkSet, because no CMYK */
if (compr==tp.COMPRESSION_JPEG) {
tp.dirUNDEFINED(tp.JPEGTables, jp->getJPEGTables().getLength(), jp->getJPEGTables()() /* , 2, "\xFF\xD9" */);
const unsigned char hvs=jp->getHVS();
if (hvs!=0x22 && phot==tp.PHOTOMETRIC_YCBCR) {
const unsigned short horiz_vert[2]={ (unsigned short)((hvs+0U)>>4),
(unsigned short)(hvs&15U) };
tp.dirSHORT(tp.YCbCrSubsampling, 2, horiz_vert);
}
if (refe) tp.dirRATIONAL(tp.ReferenceBlackWhite, 6, rats+2);
delete jp;
}
tp.dirClose();
return Rule::Applier::OK;
}
Rule::Applier out_tiff_applier = { "TIFF6", out_tiff_check_rule, out_tiff_work, 0 };
/* --- Sat Apr 20 11:32:42 CEST 2002 -- Sat Apr 20 16:48:25 CEST 2002 */
#if OBJDEP
# warning REQUIRES: crc32.o
# warning REQUIRES: crc32.o
#endif
/** Encodes 32-bit (data.length-minus), data, crc32. */
class LenCRC32Encode: public PSEncoder {
public:
/** @param maxcpl_: maximum # hex digits per line, should be even */
LenCRC32Encode(GenBuffer::Writable &out_, unsigned minus_);
virtual void vi_write(char const*buf, slen_t len);
protected:
GenBuffer::Writable &out;
unsigned minus;
/** The length is not known in advance, so we have to buffer the whole output
* in memory.
*/
SimBuffer::B sofar;
};
LenCRC32Encode::LenCRC32Encode(GenBuffer::Writable &out_, unsigned minus_): out(out_), minus(minus_), sofar("1234") {}
void LenCRC32Encode::vi_write(char const*buf, slen_t len) {
if (len==0) {
char *s=const_cast<char*>(sofar());
slen_t slen=sofar.getLength();
unsigned PTS_INT32_T crc=crc32(CRC32_INITIAL, s+4, slen-4);
slen-=minus+4;
s[0]=slen>>24; s[1]=slen>>16; s[2]=slen>>8; s[3]=slen;
s=sofar.vi_mkend(4);
s[0]=crc >>24; s[1]=crc>>16; s[2]=crc>>8; s[3]=crc;
out.vi_write(sofar(), sofar.getLength());
out.vi_write(0,0);
} else sofar.vi_write(buf, len);
}
/** PNG output (RFC 2083) */
Rule::Applier::cons_t out_png_check_rule(Rule::OutputRule* or_) {
/* Supported PNG types: /Gray1, /Gray2, /Gray4, /Gray8, /Rgb8,
* /Indexed1, /Indexed2, /Indexed4, /Indexed8.
* Unsupported PNG types: /Gray16, /Rgb16, /GrayA8, /GrayA16, /RgbA8,
* /RgbA16.
* /Indexed* supports transparency (via PNG chunk tRNS).
*/
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_PNG
) return Rule::Applier::DONT_KNOW;
bool badp=false;
/* Dat: (by policy) we don't support /Transparent or /Opaque here; the
* user should not specify such SampleFormat in the .job file.
*/
if (!cache->isGray()
&& !cache->isTransparentM()
&& !cache->isIndexed()
&& cache->SampleFormat!=Image::SF_Rgb8) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /PNG requires /Gray*, /Indexed*, /Mask, /Transparent+ or /Rgb8" << (Error*)0;
badp=true;
}
if (cache->TransferEncoding!=cache->TE_Binary) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /PNG requires /Binary" << (Error*)0;
badp=true;
}
if (cache->Compression!=Rule::Cache::CO_None && cache->Compression!=Rule::Cache::CO_ZIP) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /PNG requires /Compression/None or /ZIP" << (Error*)0;
badp=true;
}
if (cache->Predictor==cache->PR_TIFF2) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /PNG requires /Predictor >=10" << (Error*)0;
badp=true;
}
if (badp) return Rule::Applier::BAD;
/* Now we are sure about Rule::Applier::OK. */
or_->cache.WarningOK=true; /* ?? */
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_png_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
Rule::Cache *cache=&or_->cache;
char tmp[64], colortype; register char*p;
unsigned PTS_INT32_T crc;
// assert(0);
if (out_png_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
or_->doSampleFormat(sf);
Image::Sampled *img=sf->getImg();
unsigned char bpc=img->getBpc(); /* set _after_ doSampleFormat */
Image::Sampled::dimen_t wd=img->getWd(), ht=img->getHt();
GenBuffer::Writable *rp=new LenCRC32Encode(out,4);
GenBuffer::Writable *cp=PSEncoder::newFlateEncode(*rp, (cache->Compression==Rule::Cache::CO_None) ? 0: or_->cacheHints.Effort);
/* Dat: PredictorColumns, PredictorBPC and PredictorColors are forced to
* match the image, because PNG requires it.
*/
GenBuffer::Writable *pp=PSEncoder::newPredictor(
*cp,
cache->Predictor==cache->PR_None ? cache->PR_PNGNone : cache->Predictor,
bpc, /*or_->cacheHints.PredictorBPC*/
wd, /*or_->cacheHints.PredictorColumns*/
img->getCpp() /*or_->cacheHints.PredictorColors*/
);
/* Critical PNG chunks (capital 1st letter) and ancillary PNG chunks:
*
* -: zero, 1: exactly one, ?: 0 or 1, *: 0 or more, +: 1 or more
* Name Nr sam2p Ordering constraints
* IHDR 1 1 Must be first
* cHRM ? - Before PLTE and IDAT
* gAMA ? - Before PLTE and IDAT
* sBIT ? - Before PLTE and IDAT
* PLTE ? ? Before IDAT
* hIST ? - After PLTE; before IDAT
* tRNS ? ? After PLTE; before IDAT
* bKGD ? ? After PLTE; before IDAT
* pHYs ? - Before IDAT
* tIME ? - None
* tEXt * - None
* zTXt * - None
* IDAT + 1 After IHDR and PLTE, multiple IDATs must be consecutive
* IEND 1 1 Must be last
*/
/* 0..3: PNG header */
/* 8..11: IHDR chunk length */
/* 12..15: IHDR chunk name */
memcpy(tmp,"\211PNG\r\n\032\n\0\0\0\015IHDR",16); /* ASCII charset is assumed. */
p=tmp+16;
*p++=wd>>24; *p++=wd>>16; *p++=wd>>8; *p++=wd; /* 16: Width */
*p++=ht>>24; *p++=ht>>16; *p++=ht>>8; *p++=ht; /* 20: Height */
*p++=bpc; /* 24: Bit depth */
colortype=*p++=cache->isGray()?0:cache->isRGB()?2:3; /* 25: Color type */
*p++=0; /* 26: Compression method: deflate/32K (forced by PNG spec), overcoming with Effort==0 */
*p++=0; /* 27: Filter type: adaptive filtering with 5 basic PNG filter types */
*p++=0; /* 28: Non-interlaced */
crc=crc32(CRC32_INITIAL, tmp+12, 17);
*p++=crc>>24;*p++=crc>>16;*p++=crc>>8;*p++=crc; /* 29: IHDR CRC */
out.vi_write(tmp, 33);
if (colortype==3) {
bool transp=cache->isTransparentM() && PTS_dynamic_cast(Image::Indexed*,img)->getTransp()>=0;
/* unsigned ncols=PTS_dynamic_cast(Image::Indexed*,img)->getNcols(); */
unsigned ncols3=img->getRowbeg()-img->getHeadp();
p=tmp;
*p++=0; *p++=0; *p++=ncols3>>8; *p++=ncols3; /* 0: PLTE chunk length */
*p++='P'; *p++='L'; *p++='T'; *p++='E'; /* 4: PLTE chunk name */
out.vi_write(tmp, 8);
if (transp) {
PTS_dynamic_cast(Image::Indexed*,img)->makeTranspZero();
p=img->getHeadp(); p[0]=p[1]=p[2]=(char)0; /* force black backround (for consistency) */
}
out.vi_write(img->getHeadp(), ncols3);
crc=crc32(crc32(CRC32_INITIAL,tmp+4,4), img->getHeadp(), ncols3);
p=tmp; *p++=crc>>24;*p++=crc>>16;*p++=crc>>8;*p++=crc; /* 0: PLTE CRC */
out.vi_write(tmp, 4);
if (transp) {
out.vi_write("\0\0\0\1tRNS\0@\346\330f", 13); /* color 0 is transparent */
//fprintf(stderr, "tRNS: 0x%08lx", (unsigned long)crc32(CRC32_INITIAL,"tRNS\0",5)); /* 0x40e6d866 */
out.vi_write("\0\0\0\1bKGD\0\210\5\35H", 13); /* color 0 is background color */
//out.vi_write("\0\0\0\1bKGD\2\x66\x0b\x7c\x64", 13); /* color 2 is background color */
//fprintf(stderr, "bKGD: 0x%08lx", (unsigned long)crc32(CRC32_INITIAL,"bKGD\2",5)); /* 0x88051d48 */
}
}
slen_t rlenht=img->getRlen()*img->getHt();
#if 0 /* Calculates _bad_ CRC (before compression etc.) */
p=tmp;
*p++=rlenht>>24; *p++=rlenht>>16; *p++=rlenht>>8; *p++=rlenht; /* 0: IDAT chunk length */
*p++='I'; *p++='D'; *p++='A'; *p++='T'; /* 4: IDAT chunk name */
out.vi_write(tmp, 8);
if (rlenht!=0) pp->vi_write(img->getRowbeg(), rlenht);
pp->vi_write(0,0); /* flush all, write CRC */
crc=crc32(crc32(CRC32_INITIAL,tmp+4,4), img->getRowbeg(), rlenht);
p=tmp; *p++=crc>>24;*p++=crc>>16;*p++=crc>>8;*p++=crc; /* 0: IDAT CRC */
out.vi_write(tmp, 4);
#else
rp->vi_write("IDAT",4);
if (rlenht!=0) pp->vi_write(img->getRowbeg(), rlenht);
pp->vi_write(0,0); /* flush all, write CRC */
#endif
out.vi_write("\0\0\0\0IEND\256B`\202",12); /* IEND chunk */
delete pp;
delete cp;
delete rp;
return Rule::Applier::OK;
}
Rule::Applier out_png_applier = { "PNG", out_png_check_rule, out_png_work, 0 };
/* --- Sat Jun 15 19:30:41 CEST 2002 */
/** BMP RLE compression, type 1. Seems to be optimal.
* @param dst buffer with enough place for compressed data. The caller must
* pre-allocate >=(send-sbeg)+(send-sbeg+128)*2/255.
* @param sbeg first raw data char to compress
* @param send char to finish compression just before
* @return dst+(number of characters occupied by compressed data)
*
* in-memory implementation Sun Jun 23 20:02:41 CEST 2002
*/
static char *bmp_compress1_row(char *dst, char const *sbeg, char const *send) {
#if 0
# define BMP_ASSERT(x) assert(x)
#else
# define BMP_ASSERT(x)
#endif
#undef BUF
#define BUF(x) (*(x))
#undef PUTCH__
#define PUTCH__(c) (*dst++=(c))
char c, c2;
char const *beg, *end, *rend, *q, *r, *best;
signed bestca, rca; /* both must fit into -255..255 */
slen_t frl, efrl;
int ci;
// bool oddp;
beg=sbeg;
end=(send-sbeg>255) ? beg+255 : send;
while (beg!=end) { /* there is still unprocessed data in the buffer */
c=BUF(beg++);
if (beg==end) { PUTCH__(1); PUTCH__(c); break; } /* last char */
if (c==BUF(beg)) { /* sure r chunk */
ci=2; beg++;
rep:
while (beg!=end && c==BUF(beg)) { beg++; ci++; }
PUTCH__(ci); PUTCH__(c); /* r chunk */
} else { /* possible c chunk */
rend=end;
BMP_ASSERT(end-beg<=254);
if (end!=send) { /* read an extra char as the terminator of the last run-length of c, buf:0..254 */
end++;
BMP_ASSERT(end-beg==255); /* buffer is full (255 chars) */
}
best=r=beg;
bestca=rca=-1; /* best and current advantage of c over r */
while (r!=rend) { /* c chunk should stop only at run-length boundaries */
BMP_ASSERT(-255<=rca && rca<=255);
BMP_ASSERT(-255<=bestca && bestca<=255);
q=r; r=q+1; ci=1; while (r!=end && BUF(r)==BUF(q)) { r++; ci++; }
if (r==end && end!=rend) break;
if (((r-beg)&1)==0) { /* odd (!) copy length */
rca+=3-ci;
if (rca<=bestca-1) { r--; break; } /* fix-5 (instead of rule-4): `xyz|bbbbb|xyz|', `abcdef|gggggggg|abababababab|', `abcdef|ggg|hhh|ggg|hhh|ggg|hhh|ggg|hhh|abababababab|' */
if (bestca<rca) { bestca=rca; best=r-1; } /* make c as short as possible */
rca--;
} else { /* even copy length */
rca+=2-ci;
if (rca<=bestca-2) break; /* fix-5 (instead of rule-4): `xyz|bbbbb|xyz|', `abcdef|gggggggg|abababababab|', `abcdef|ggg|hhh|ggg|hhh|ggg|hhh|ggg|hhh|abababababab|' */
if (bestca<rca) { bestca=rca; best=r; } /* make c as short as possible */
}
}
BMP_ASSERT(-255<=rca && rca<=255);
BMP_ASSERT(-255<=bestca && bestca<=255);
if (bestca<=0 /* no possible positive advantage */
|| best-beg<=1 /* 1: c is one char, plus 1 char in buf. Imp: ==1?! */
) { ci=1; goto rep; }
r=best; /* Imp: get rid of this assignment */
BMP_ASSERT(beg!=r);
BMP_ASSERT(((r-beg)&1)==1); /* even copy length */
if (end==r) { /* no followers, last chunk */
/* BMP_ASSERT(had_eof); */
// oddp=(1+(r-beg)&1)==1;
PUTCH__(0);
PUTCH__((r-beg)+1);
PUTCH__(c);
while (beg!=r) { PUTCH__(BUF(beg)); beg++; } /* emit c chunk */
// if (oddp) PUTCH__(0); /* Imp: padding breaks optimality */
} else {
BMP_ASSERT(r!=end);
/* BMP_ASSERT(r!=rend); */ /* r==rend is possible here */
c2=BUF(r); frl=1; q=r+1;
while (q!=end && c2==BUF(q)) { q++; frl++; } /* count follower run length */
efrl=frl; ci=-2; if (q==end) { /* Imp: get rid of -2 (-2 -> -1) */
BMP_ASSERT(q==end);
while ((ci=(q==send)?-1:(unsigned char)*q++)!=-1 && (char)ci==c2) efrl++;
}
/* printf("clen=%u\n", clen); */
if (1+(r>beg ? r-beg : 256+beg-r)<255 && efrl>=256 && efrl%255==1) { r++; efrl--; } /* make the c chunk one char longer if appropriate */
// oddp=(1+(r-beg)&1)==1;
PUTCH__(0);
PUTCH__(1+(r-beg));
PUTCH__(c);
while (beg!=r) { PUTCH__(BUF(beg)); beg++; } /* emit c chunk */
// if (oddp) PUTCH__(0); /* Imp: padding breaks optimality */
beg=q; /* remove beginning of the r chunk from the buffer */
if (ci>=0) { beg--; BMP_ASSERT((unsigned char)BUF(beg)==ci); }
while (efrl>=255) { PUTCH__('\377'); PUTCH__(c2); efrl-=255; } /* emit full r chunks */
if (efrl>=2) { /* emit last r chunk */
PUTCH__(efrl); PUTCH__(c2);
} else if (efrl!=0) {
BMP_ASSERT(efrl==1);
beg--; /* leave a single instance of c2 in beginning of the buffer */
BMP_ASSERT(BUF(beg)==c2);
}
} /* IF c chunk has followers */
} /* IF r or c chunk */
end=(send-beg>255) ? beg+255 : send;
} /* WHILE main loop */
return dst;
#undef BUF
#undef PUTCH__
}
/** Windows Bitmap BMP output */
Rule::Applier::cons_t out_bmp_check_rule(Rule::OutputRule* or_) {
/* Supported BMP types: /Rgb8,
* /Indexed1, /Indexed4, /Indexed8.
*/
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_BMP
) return Rule::Applier::DONT_KNOW;
bool badp=false;
/* Dat: (by policy) we don't support /Transparent or /Opaque here; the
* user should not specify such SampleFormat in the .job file.
*/
if (cache->SampleFormat!=Image::SF_Indexed1
/* Dat: /Indexed2 is not supported by /BMP */
&& cache->SampleFormat!=Image::SF_Indexed4
&& cache->SampleFormat!=Image::SF_Indexed8
&& cache->SampleFormat!=Image::SF_Rgb8) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /BMP requires /Indexed1, /Indexed4, /Indexed8 or /Rgb8" << (Error*)0;
badp=true;
}
if (cache->TransferEncoding!=cache->TE_Binary) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /BMP requires /Binary" << (Error*)0;
badp=true;
}
if (cache->Compression!=Rule::Cache::CO_None && cache->Compression!=Rule::Cache::CO_RLE) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /BMP requires /Compression/None or /RLE" << (Error*)0;
badp=true;
}
if (cache->Compression==Rule::Cache::CO_RLE && cache->SampleFormat!=Image::SF_Indexed8) {
/* !! Imp: Implement compr==2 for /Indexed4 */
Error::sev(Error::WARNING_DEFER) << "check_rule: /BMP/RLE requires /Indexed8" << (Error*)0;
badp=true;
}
if (cache->hasPredictor()) {
Error::sev(Error::WARNING_DEFER) << "check_rule: /BMP requires /Predictor 1" << (Error*)0;
badp=true;
}
if (badp) return Rule::Applier::BAD;
/* Now we are sure about Rule::Applier::OK. */
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_bmp_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
if (out_bmp_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
or_->doSampleFormat(sf);
Image::Sampled *img=sf->getImg();
Image::Sampled::dimen_t wd=img->getWd(), ht=img->getHt(), htc=ht;
slen_t rlen=img->getRlen();
char const*pend=img->getRowbeg()+rlen*ht;
char const*palbeg=img->getHeadp(), *palend=img->getRowbeg();
slen_t ncols4=(palend-palbeg)/3*4;
unsigned biCompr=or_->cache.Compression!=or_->cache.CO_RLE ? 0
: or_->cache.SampleFormat==Image::SF_Indexed4 ? 2 : 1;
SimBuffer::B data;
slen_t crowsize=2+ rlen+(rlen+128)*2/255; /* !! Imp: real upper bound? */
char *crow=new char[crowsize];
/* !! GIMP compatibility */
if (or_->cache.Compression==or_->cache.CO_RLE) {
/* Imp: bmp_compress2_row */
char *crow2;
while (htc--!=0) { /* BMP stores rows from down */
crow2=bmp_compress1_row(crow, pend-rlen, pend);
#if 0
crow2=crow;
*crow2++=10;
*crow2++=1;
*crow2++=10;
*crow2++=2;
*crow2++=10;
*crow2++=3;
*crow2++=10;
*crow2++=4;
*crow2++=10;
*crow2++=5;
*crow2++=10;
*crow2++=6;
#endif
assert((slen_t)(crow2-crow)<=crowsize-2);
pend-=rlen;
*crow2++='\0'; *crow2++='\0'; /* signal end of compressed data row */
/* fprintf(stderr, "htc=%u cl=%u\n", htc, crow2-crow); */
data.vi_write(crow, crow2-crow);
}
data.vi_write("\0\1", 2); /* signal end of bitmap */
} else {
unsigned pad=(4-(rlen&3))&3; /* Dat: pad rows to 32 bits */
if (or_->cache.SampleFormat==Image::SF_Rgb8) { /* SWAP RGB values */
/* BUGFIX at Thu Dec 12 21:36:57 CET 2002 */
char *buf=new char[rlen];
while (htc--!=0) { /* BMP stores rows from down */
char const*pxend=pend, *px=pend-=rlen;
char *q=buf;
while (px!=pxend) {
*q++=px[2];
*q++=px[1];
*q++=px[3];
px+=3;
}
data.vi_write(buf, rlen);
if (pad!=0) data.vi_write("\0\0\0", pad);
}
delete [] buf;
} else {
while (htc--!=0) { /* BMP stores rows from down */
data.vi_write(pend-=rlen, rlen);
if (pad!=0) data.vi_write("\0\0\0", pad);
}
}
}
assert(pend==img->getRowbeg());
delete [] crow;
/* Now data is ready. */
char *bmphead=new char[54+ncols4+data.getLength()], *p=bmphead;
*p++='B'; *p++='M'; /* magic number header */
lf32(p, 54+ncols4+data.getLength()); /* bfSize */
lf32(p, 0); /* zzHotX, zzHotY */
lf32(p, 54+ncols4); /* bfOffs */
lf32(p, 40); /* biSize==40 => Windows 3.x style BMP file */
lf32(p, wd); /* biWidth */
lf32(p, ht); /* biHeight */
lf16(p, 1); /* biPlanes, must be 1 */
lf16(p, img->getBpc()*img->getCpp()); /* biBitCnt: bits per pixel (1, 4, 8: /Indexed*; 24: /Rgb8) */
lf32(p, biCompr); /* biCompr */
lf32(p, 0); /* biSizeIm */
lf32(p, 0); /* biXPels */
lf32(p, 0); /* biYPels */
lf32(p, (palend-palbeg)/3); /* biClrUsed */
lf32(p, (palend-palbeg)/3); /* biClrImp */
/* vvv Now comes the palette (zero length for /Rgb) */
while (palbeg!=palend) {
*p++=palbeg[2]; *p++=palbeg[1]; *p++=palbeg[0]; *p++='\0';
palbeg+=3;
}
assert((slen_t)(p-bmphead)==54+ncols4);
out.vi_write(bmphead, p-bmphead); delete [] bmphead;
#if 0
out.vi_write("XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX",
100);
out.vi_write("XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX",
100);
out.vi_write("XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX",
100);
out.vi_write("XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX",
100);
out.vi_write("XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX",
100);
#endif
out << data;
out.vi_write(0,0); /* signal EOF */
return Rule::Applier::OK;
}
Rule::Applier out_bmp_applier = { "BMP", out_bmp_check_rule, out_bmp_work, 0 };
/* --- Sun Mar 24 13:48:57 CET 2002 */
/** The Empty applier produces an empty output file. */
Rule::Applier::cons_t out_empty_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_Empty
) return Rule::Applier::DONT_KNOW;
return Rule::Applier::OK;
}
Rule::Applier::cons_t out_empty_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
(void)out;
(void)sf;
if (out_empty_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
return Rule::Applier::OK;
}
Rule::Applier out_empty_applier = { "Empty", out_empty_check_rule, out_empty_work, 0 };
/* --- Sun Mar 24 13:48:57 CET 2002 */
/** The Meta applier writes some meta-information into the output file */
Rule::Applier::cons_t out_meta_check_rule(Rule::OutputRule* or_) {
Rule::Cache *cache=&or_->cache;
if (cache->FileFormat!=cache->FF_Meta
) return Rule::Applier::DONT_KNOW;
// return Rule::Applier::OK; /* Imp: implement the applier */
return Rule::Applier::BAD;
}
Rule::Applier::cons_t out_meta_work(GenBuffer::Writable& out, Rule::OutputRule*or_, Image::SampledInfo *sf) {
(void)out;
(void)sf;
if (out_meta_check_rule(or_)!=Rule::Applier::OK) return Rule::Applier::DONT_KNOW;
/* Imp: real content here */
return Rule::Applier::OK;
}
Rule::Applier out_meta_applier = { "Meta", out_meta_check_rule, out_meta_work, 0 };
/* __END__ */