123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708 |
- //---------------------------------------------------------------------------------
- //
- // Little Color Management System
- // Copyright (c) 1998-2022 Marti Maria Saguer
- //
- // Permission is hereby granted, free of charge, to any person obtaining
- // a copy of this software and associated documentation files (the "Software"),
- // to deal in the Software without restriction, including without limitation
- // the rights to use, copy, modify, merge, publish, distribute, sublicense,
- // and/or sell copies of the Software, and to permit persons to whom the Software
- // is furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO
- // THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- //
- //---------------------------------------------------------------------------------
- //
- #include "utils.h"
- #include "tiffio.h"
- // ------------------------------------------------------------------------
- static TIFF *Tiff1, *Tiff2, *TiffDiff;
- static const char* TiffDiffFilename;
- static const char* CGATSout;
- typedef struct {
- double n, x, x2;
- double Min, Peak;
- } STAT, *LPSTAT;
- static STAT ColorantStat[4];
- static STAT EuclideanStat;
- static STAT ColorimetricStat;
- static uint16 Channels;
- static cmsHPROFILE hLab;
- static
- void ConsoleWarningHandler(const char* module, const char* fmt, va_list ap)
- {
- char e[512] = { '\0' };
- if (module != NULL)
- strcat(strcpy(e, module), ": ");
- vsprintf(e+strlen(e), fmt, ap);
- strcat(e, ".");
- if (Verbose) {
- fprintf(stderr, "\nWarning");
- fprintf(stderr, " %s\n", e);
- fflush(stderr);
- }
- }
- static
- void ConsoleErrorHandler(const char* module, const char* fmt, va_list ap)
- {
- char e[512] = { '\0' };
- if (module != NULL)
- strcat(strcpy(e, module), ": ");
- vsprintf(e+strlen(e), fmt, ap);
- strcat(e, ".");
- fprintf(stderr, "\nError");
- fprintf(stderr, " %s\n", e);
- fflush(stderr);
- }
- static
- void Help()
- {
- fprintf(stderr, "Little CMS TIFF compare utility. v1.1\n\n");
- fprintf(stderr, "usage: tiffdiff [flags] input.tif output.tif\n");
- fprintf(stderr, "\nflags:\n\n");
- fprintf(stderr, "-o<tiff> - Output TIFF file\n");
- fprintf(stderr, "-g<CGATS> - Output results in CGATS file\n");
-
- fprintf(stderr, "\n");
- fprintf(stderr, "-v - Verbose (show warnings)\n");
- fprintf(stderr, "-h - This help\n");
- fflush(stderr);
- exit(0);
- }
- // The toggles stuff
- static
- void HandleSwitches(int argc, char *argv[])
- {
- int s;
-
- while ((s=xgetopt(argc,argv,"o:O:hHvVg:G:")) != EOF) {
- switch (s) {
- case 'v':
- case 'V':
- Verbose = TRUE;
- break;
- case 'o':
- case 'O':
- TiffDiffFilename = xoptarg;
- break;
-
- case 'H':
- case 'h':
- Help();
- break;
- case 'g':
- case 'G':
- CGATSout = xoptarg;
- break;
- default:
- FatalError("Unknown option - run without args to see valid ones");
- }
- }
- }
- static
- void ClearStatistics(LPSTAT st)
- {
- st ->n = st ->x = st->x2 = st->Peak = 0;
- st ->Min = 1E10;
-
- }
- static
- void AddOnePixel(LPSTAT st, double dE)
- {
-
- st-> x += dE; st ->x2 += (dE * dE); st->n += 1.0;
- if (dE > st ->Peak) st ->Peak = dE;
- if (dE < st ->Min) st ->Min= dE;
- }
- static
- double Std(LPSTAT st)
- {
- return sqrt((st->n * st->x2 - st->x * st->x) / (st->n*(st->n-1)));
- }
-
- static
- double Mean(LPSTAT st)
- {
- return st ->x/st ->n;
- }
- // Build up the pixeltype descriptor
- static
- cmsUInt32Number GetInputPixelType(TIFF *Bank)
- {
- uint16 Photometric, bps, spp, extra, PlanarConfig, *info;
- uint16 Compression, reverse = 0;
- int ColorChannels, IsPlanar = 0, pt = 0;
- TIFFGetField(Bank, TIFFTAG_PHOTOMETRIC, &Photometric);
- TIFFGetFieldDefaulted(Bank, TIFFTAG_BITSPERSAMPLE, &bps);
- if (bps == 1)
- FatalError("Sorry, bilevel TIFFs has nothig to do with ICC profiles");
- if (bps != 8 && bps != 16)
- FatalError("Sorry, 8 or 16 bits per sample only");
- TIFFGetFieldDefaulted(Bank, TIFFTAG_SAMPLESPERPIXEL, &spp);
- TIFFGetFieldDefaulted(Bank, TIFFTAG_PLANARCONFIG, &PlanarConfig);
- switch (PlanarConfig)
- {
- case PLANARCONFIG_CONTIG: IsPlanar = 0; break;
- case PLANARCONFIG_SEPARATE: FatalError("Planar TIFF are not supported");
- default:
- FatalError("Unsupported planar configuration (=%d) ", (int) PlanarConfig);
- }
- // If Samples per pixel == 1, PlanarConfiguration is irrelevant and need
- // not to be included.
- if (spp == 1) IsPlanar = 0;
- // Any alpha?
- TIFFGetFieldDefaulted(Bank, TIFFTAG_EXTRASAMPLES, &extra, &info);
-
- ColorChannels = spp - extra;
- switch (Photometric) {
- case PHOTOMETRIC_MINISWHITE:
-
- reverse = 1;
- case PHOTOMETRIC_MINISBLACK:
-
- pt = PT_GRAY;
- break;
- case PHOTOMETRIC_RGB:
-
- pt = PT_RGB;
- break;
- case PHOTOMETRIC_PALETTE:
-
- FatalError("Sorry, palette images not supported (at least on this version)");
- case PHOTOMETRIC_SEPARATED:
- pt = PixelTypeFromChanCount(ColorChannels);
- break;
- case PHOTOMETRIC_YCBCR:
- TIFFGetField(Bank, TIFFTAG_COMPRESSION, &Compression);
- {
- uint16 subx, suby;
- pt = PT_YCbCr;
- TIFFGetFieldDefaulted(Bank, TIFFTAG_YCBCRSUBSAMPLING, &subx, &suby);
- if (subx != 1 || suby != 1)
- FatalError("Sorry, subsampled images not supported");
- }
- break;
- case 9:
- case PHOTOMETRIC_CIELAB:
- pt = PT_Lab;
- break;
-
- case PHOTOMETRIC_LOGLUV: /* CIE Log2(L) (u',v') */
- TIFFSetField(Bank, TIFFTAG_SGILOGDATAFMT, SGILOGDATAFMT_16BIT);
- pt = PT_YUV; // *ICCSpace = icSigLuvData;
- bps = 16; // 16 bits forced by LibTiff
- break;
- default:
- FatalError("Unsupported TIFF color space (Photometric %d)", Photometric);
- }
- // Convert bits per sample to bytes per sample
- bps >>= 3;
- return (COLORSPACE_SH(pt)|PLANAR_SH(IsPlanar)|EXTRA_SH(extra)|CHANNELS_SH(ColorChannels)|BYTES_SH(bps)|FLAVOR_SH(reverse));
- }
- static
- cmsUInt32Number OpenEmbedded(TIFF* tiff, cmsHPROFILE* PtrProfile, cmsHTRANSFORM* PtrXform)
- {
- cmsUInt32Number EmbedLen, dwFormat = 0;
- cmsUInt8Number* EmbedBuffer;
-
- *PtrProfile = NULL;
- *PtrXform = NULL;
- if (TIFFGetField(tiff, TIFFTAG_ICCPROFILE, &EmbedLen, &EmbedBuffer)) {
- *PtrProfile = cmsOpenProfileFromMem(EmbedBuffer, EmbedLen);
-
- if (Verbose) {
-
- fprintf(stdout, "Embedded profile found:\n");
- PrintProfileInformation(*PtrProfile);
-
- }
- dwFormat = GetInputPixelType(tiff);
- *PtrXform = cmsCreateTransform(*PtrProfile, dwFormat,
- hLab, TYPE_Lab_DBL, INTENT_RELATIVE_COLORIMETRIC, 0);
- }
- return dwFormat;
- }
- static
- size_t PixelSize(cmsUInt32Number dwFormat)
- {
- return T_BYTES(dwFormat) * (T_CHANNELS(dwFormat) + T_EXTRA(dwFormat));
- }
- static
- int CmpImages(TIFF* tiff1, TIFF* tiff2, TIFF* diff)
- {
- cmsUInt8Number* buf1, *buf2, *buf3=NULL;
- int row, cols, imagewidth = 0, imagelength = 0;
- uint16 Photometric;
- double dE = 0;
- double dR, dG, dB, dC, dM, dY, dK;
- int rc = 0;
- cmsHPROFILE hProfile1 = 0, hProfile2 = 0;
- cmsHTRANSFORM xform1 = 0, xform2 = 0;
- cmsUInt32Number dwFormat1, dwFormat2;
-
- TIFFGetField(tiff1, TIFFTAG_PHOTOMETRIC, &Photometric);
- TIFFGetField(tiff1, TIFFTAG_IMAGEWIDTH, &imagewidth);
- TIFFGetField(tiff1, TIFFTAG_IMAGELENGTH, &imagelength);
- TIFFGetField(tiff1, TIFFTAG_SAMPLESPERPIXEL, &Channels);
-
- dwFormat1 = OpenEmbedded(tiff1, &hProfile1, &xform1);
- dwFormat2 = OpenEmbedded(tiff2, &hProfile2, &xform2);
-
-
-
- buf1 = (cmsUInt8Number*)_TIFFmalloc(TIFFScanlineSize(tiff1));
- buf2 = (cmsUInt8Number*)_TIFFmalloc(TIFFScanlineSize(tiff2));
-
- if (diff) {
-
- TIFFSetField(diff, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_MINISBLACK);
- TIFFSetField(diff, TIFFTAG_COMPRESSION, COMPRESSION_NONE);
- TIFFSetField(diff, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
- TIFFSetField(diff, TIFFTAG_IMAGEWIDTH, imagewidth);
- TIFFSetField(diff, TIFFTAG_IMAGELENGTH, imagelength);
- TIFFSetField(diff, TIFFTAG_SAMPLESPERPIXEL, 1);
- TIFFSetField(diff, TIFFTAG_BITSPERSAMPLE, 8);
-
- buf3 = (cmsUInt8Number*)_TIFFmalloc(TIFFScanlineSize(diff));
- }
-
- for (row = 0; row < imagelength; row++) {
- if (TIFFReadScanline(tiff1, buf1, row, 0) < 0) goto Error;
- if (TIFFReadScanline(tiff2, buf2, row, 0) < 0) goto Error;
-
- for (cols = 0; cols < imagewidth; cols++) {
-
- switch (Photometric) {
- case PHOTOMETRIC_MINISWHITE:
- case PHOTOMETRIC_MINISBLACK:
- dE = fabs(buf2[cols] - buf1[cols]);
-
- AddOnePixel(&ColorantStat[0], dE);
- AddOnePixel(&EuclideanStat, dE);
- break;
- case PHOTOMETRIC_RGB:
-
- {
- int index = 3 * cols;
- dR = fabs(buf2[index+0] - buf1[index+0]);
- dG = fabs(buf2[index+1] - buf1[index+1]);
- dB = fabs(buf2[index+2] - buf1[index+2]);
- dE = sqrt(dR * dR + dG * dG + dB * dB) / sqrt(3.);
- }
- AddOnePixel(&ColorantStat[0], dR);
- AddOnePixel(&ColorantStat[1], dG);
- AddOnePixel(&ColorantStat[2], dB);
- AddOnePixel(&EuclideanStat, dE);
- break;
- case PHOTOMETRIC_SEPARATED:
-
- {
- int index = 4 * cols;
- dC = fabs(buf2[index+0] - buf1[index+0]);
- dM = fabs(buf2[index+1] - buf1[index+1]);
- dY = fabs(buf2[index+2] - buf1[index+2]);
- dK = fabs(buf2[index+3] - buf1[index+3]);
- dE = sqrt(dC * dC + dM * dM + dY * dY + dK * dK) / 2.;
- }
- AddOnePixel(&ColorantStat[0], dC);
- AddOnePixel(&ColorantStat[1], dM);
- AddOnePixel(&ColorantStat[2], dY);
- AddOnePixel(&ColorantStat[3], dK);
- AddOnePixel(&EuclideanStat, dE);
- break;
-
- default:
- FatalError("Unsupported channels: %d", Channels);
- }
-
- if (xform1 && xform2) {
-
- cmsCIELab Lab1, Lab2;
- size_t index1 = cols * PixelSize(dwFormat1);
- size_t index2 = cols * PixelSize(dwFormat2);
- cmsDoTransform(xform1, &buf1[index1], &Lab1, 1);
- cmsDoTransform(xform2, &buf2[index2], &Lab2, 1);
- dE = cmsDeltaE(&Lab1, &Lab2);
- AddOnePixel(&ColorimetricStat, dE);
- }
- if (diff) {
- buf3[cols] = (cmsUInt8Number) floor(dE + 0.5);
- }
- }
- if (diff) {
- if (TIFFWriteScanline(diff, buf3, row, 0) < 0) goto Error;
- }
-
- }
- rc = 1;
- Error:
-
- if (hProfile1) cmsCloseProfile(hProfile1);
- if (hProfile2) cmsCloseProfile(hProfile2);
- if (xform1) cmsDeleteTransform(xform1);
- if (xform2) cmsDeleteTransform(xform2);
- _TIFFfree(buf1); _TIFFfree(buf2);
- if (diff) {
- TIFFWriteDirectory(diff);
- if (buf3 != NULL) _TIFFfree(buf3);
- }
- return rc;
- }
- static
- void AssureShortTagIs(TIFF* tif1, TIFF* tiff2, int tag, int Val, const char* Error)
- {
- uint16 v1;
-
- if (!TIFFGetField(tif1, tag, &v1)) goto Err;
- if (v1 != Val) goto Err;
- if (!TIFFGetField(tiff2, tag, &v1)) goto Err;
- if (v1 != Val) goto Err;
- return;
- Err:
- FatalError("%s is not proper", Error);
- }
- static
- int CmpShortTag(TIFF* tif1, TIFF* tif2, int tag)
- {
- uint16 v1, v2;
- if (!TIFFGetField(tif1, tag, &v1)) return 0;
- if (!TIFFGetField(tif2, tag, &v2)) return 0;
- return v1 == v2;
- }
- static
- int CmpLongTag(TIFF* tif1, TIFF* tif2, int tag)
- {
- uint32 v1, v2;
- if (!TIFFGetField(tif1, tag, &v1)) return 0;
- if (!TIFFGetField(tif2, tag, &v2)) return 0;
- return v1 == v2;
- }
- static
- void EqualShortTag(TIFF* tif1, TIFF* tif2, int tag, const char* Error)
- {
- if (!CmpShortTag(tif1, tif2, tag))
- FatalError("%s is different", Error);
- }
- static
- void EqualLongTag(TIFF* tif1, TIFF* tif2, int tag, const char* Error)
- {
- if (!CmpLongTag(tif1, tif2, tag))
- FatalError("%s is different", Error);
- }
- static
- void AddOneCGATSRow(cmsHANDLE hIT8, char *Name, LPSTAT st)
- {
- double Per100 = 100.0 * ((255.0 - Mean(st)) / 255.0);
- cmsIT8SetData(hIT8, Name, "SAMPLE_ID", Name);
- cmsIT8SetDataDbl(hIT8, Name, "PER100_EQUAL", Per100);
- cmsIT8SetDataDbl(hIT8, Name, "MEAN_DE", Mean(st));
- cmsIT8SetDataDbl(hIT8, Name, "STDEV_DE", Std(st));
- cmsIT8SetDataDbl(hIT8, Name, "MIN_DE", st ->Min);
- cmsIT8SetDataDbl(hIT8, Name, "MAX_DE", st ->Peak);
- }
- static
- void CreateCGATS(const char* TiffName1, const char* TiffName2)
- {
- cmsHANDLE hIT8 = cmsIT8Alloc(0);
- time_t ltime;
- char Buffer[256];
- cmsIT8SetSheetType(hIT8, "TIFFDIFF");
-
-
- sprintf(Buffer, "Differences between %s and %s", TiffName1, TiffName2);
-
- cmsIT8SetComment(hIT8, Buffer);
- cmsIT8SetPropertyStr(hIT8, "ORIGINATOR", "TIFFDIFF");
- time( <ime );
- strcpy(Buffer, ctime(<ime));
- Buffer[strlen(Buffer)-1] = 0; // Remove the nasty "\n"
- cmsIT8SetPropertyStr(hIT8, "CREATED", Buffer);
- cmsIT8SetComment(hIT8, " ");
- cmsIT8SetPropertyDbl(hIT8, "NUMBER_OF_FIELDS", 6);
-
-
- cmsIT8SetDataFormat(hIT8, 0, "SAMPLE_ID");
- cmsIT8SetDataFormat(hIT8, 1, "PER100_EQUAL");
- cmsIT8SetDataFormat(hIT8, 2, "MEAN_DE");
- cmsIT8SetDataFormat(hIT8, 3, "STDEV_DE");
- cmsIT8SetDataFormat(hIT8, 4, "MIN_DE");
- cmsIT8SetDataFormat(hIT8, 5, "MAX_DE");
-
- switch (Channels) {
- case 1:
- cmsIT8SetPropertyDbl(hIT8, "NUMBER_OF_SETS", 3);
- AddOneCGATSRow(hIT8, "GRAY_PLANE", &ColorantStat[0]);
- break;
- case 3:
- cmsIT8SetPropertyDbl(hIT8, "NUMBER_OF_SETS", 5);
- AddOneCGATSRow(hIT8, "R_PLANE", &ColorantStat[0]);
- AddOneCGATSRow(hIT8, "G_PLANE", &ColorantStat[1]);
- AddOneCGATSRow(hIT8, "B_PLANE", &ColorantStat[2]);
- break;
-
-
- case 4:
- cmsIT8SetPropertyDbl(hIT8, "NUMBER_OF_SETS", 6);
- AddOneCGATSRow(hIT8, "C_PLANE", &ColorantStat[0]);
- AddOneCGATSRow(hIT8, "M_PLANE", &ColorantStat[1]);
- AddOneCGATSRow(hIT8, "Y_PLANE", &ColorantStat[2]);
- AddOneCGATSRow(hIT8, "K_PLANE", &ColorantStat[3]);
- break;
-
- default: FatalError("Internal error: Bad ColorSpace");
- }
- AddOneCGATSRow(hIT8, "EUCLIDEAN", &EuclideanStat);
- AddOneCGATSRow(hIT8, "COLORIMETRIC", &ColorimetricStat);
- cmsIT8SaveToFile(hIT8, CGATSout);
- cmsIT8Free(hIT8);
- }
- int main(int argc, char* argv[])
- {
- int i;
- Tiff1 = Tiff2 = TiffDiff = NULL;
- InitUtils("tiffdiff");
- HandleSwitches(argc, argv);
- if ((argc - xoptind) != 2) {
- Help();
- }
-
- TIFFSetErrorHandler(ConsoleErrorHandler);
- TIFFSetWarningHandler(ConsoleWarningHandler);
- Tiff1 = TIFFOpen(argv[xoptind], "r");
- if (Tiff1 == NULL) FatalError("Unable to open '%s'", argv[xoptind]);
- Tiff2 = TIFFOpen(argv[xoptind+1], "r");
- if (Tiff2 == NULL) FatalError("Unable to open '%s'", argv[xoptind+1]);
-
- if (TiffDiffFilename) {
- TiffDiff = TIFFOpen(TiffDiffFilename, "w");
- if (TiffDiff == NULL) FatalError("Unable to create '%s'", TiffDiffFilename);
- }
-
- AssureShortTagIs(Tiff1, Tiff2, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG, "Planar Config");
- AssureShortTagIs(Tiff1, Tiff2, TIFFTAG_BITSPERSAMPLE, 8, "8 bit per sample");
- EqualLongTag(Tiff1, Tiff2, TIFFTAG_IMAGEWIDTH, "Image width");
- EqualLongTag(Tiff1, Tiff2, TIFFTAG_IMAGELENGTH, "Image length");
-
- EqualShortTag(Tiff1, Tiff2, TIFFTAG_SAMPLESPERPIXEL, "Samples per pixel");
- hLab = cmsCreateLab4Profile(NULL);
- ClearStatistics(&EuclideanStat);
- for (i=0; i < 4; i++)
- ClearStatistics(&ColorantStat[i]);
- if (!CmpImages(Tiff1, Tiff2, TiffDiff))
- FatalError("Error comparing images");
- if (CGATSout) {
- CreateCGATS(argv[xoptind], argv[xoptind+1]);
- }
- else {
- double Per100 = 100.0 * ((255.0 - Mean(&EuclideanStat)) / 255.0);
- printf("Digital counts %g%% equal. mean %g, min %g, max %g, Std %g\n", Per100, Mean(&EuclideanStat),
- EuclideanStat.Min,
- EuclideanStat.Peak,
- Std(&EuclideanStat));
- if (ColorimetricStat.n > 0) {
- Per100 = 100.0 * ((255.0 - Mean(&ColorimetricStat)) / 255.0);
- printf("dE Colorimetric %g%% equal. mean %g, min %g, max %g, Std %g\n", Per100, Mean(&ColorimetricStat),
- ColorimetricStat.Min,
- ColorimetricStat.Peak,
- Std(&ColorimetricStat));
- }
-
- }
- if (hLab) cmsCloseProfile(hLab);
- if (Tiff1) TIFFClose(Tiff1);
- if (Tiff2) TIFFClose(Tiff2);
- if (TiffDiff) TIFFClose(TiffDiff);
- return 0;
- }
|