Blame tools/rgb2ycbcr.c

Packit 7838c8
/* $Id: rgb2ycbcr.c,v 1.17 2016-08-15 21:26:56 erouault Exp $ */
Packit 7838c8
Packit 7838c8
/*
Packit 7838c8
 * Copyright (c) 1991-1997 Sam Leffler
Packit 7838c8
 * Copyright (c) 1991-1997 Silicon Graphics, Inc.
Packit 7838c8
 *
Packit 7838c8
 * Permission to use, copy, modify, distribute, and sell this software and 
Packit 7838c8
 * its documentation for any purpose is hereby granted without fee, provided
Packit 7838c8
 * that (i) the above copyright notices and this permission notice appear in
Packit 7838c8
 * all copies of the software and related documentation, and (ii) the names of
Packit 7838c8
 * Sam Leffler and Silicon Graphics may not be used in any advertising or
Packit 7838c8
 * publicity relating to the software without the specific, prior written
Packit 7838c8
 * permission of Sam Leffler and Silicon Graphics.
Packit 7838c8
 * 
Packit 7838c8
 * THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND, 
Packit 7838c8
 * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY 
Packit 7838c8
 * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  
Packit 7838c8
 * 
Packit 7838c8
 * IN NO EVENT SHALL SAM LEFFLER OR SILICON GRAPHICS BE LIABLE FOR
Packit 7838c8
 * ANY SPECIAL, INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND,
Packit 7838c8
 * OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
Packit 7838c8
 * WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF 
Packit 7838c8
 * LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE 
Packit 7838c8
 * OF THIS SOFTWARE.
Packit 7838c8
 */
Packit 7838c8
Packit 7838c8
#include "tif_config.h"
Packit 7838c8
Packit 7838c8
#include <stdio.h>
Packit 7838c8
#include <string.h>
Packit 7838c8
#include <stdlib.h>
Packit 7838c8
Packit 7838c8
#ifdef HAVE_UNISTD_H
Packit 7838c8
# include <unistd.h>
Packit 7838c8
#endif
Packit 7838c8
Packit 7838c8
#ifdef NEED_LIBPORT
Packit 7838c8
# include "libport.h"
Packit 7838c8
#endif
Packit 7838c8
Packit 7838c8
#include "tiffiop.h"
Packit 7838c8
#include "tiffio.h"
Packit 7838c8
Packit 7838c8
#define	streq(a,b)	(strcmp(a,b) == 0)
Packit 7838c8
#define	CopyField(tag, v) \
Packit 7838c8
    if (TIFFGetField(in, tag, &v)) TIFFSetField(out, tag, v)
Packit 7838c8
Packit 7838c8
#ifndef howmany
Packit 7838c8
#define	howmany(x, y)	(((x)+((y)-1))/(y))
Packit 7838c8
#endif
Packit 7838c8
#define	roundup(x, y)	(howmany(x,y)*((uint32)(y)))
Packit 7838c8
Packit 7838c8
#define	LumaRed		ycbcrCoeffs[0]
Packit 7838c8
#define	LumaGreen	ycbcrCoeffs[1]
Packit 7838c8
#define	LumaBlue	ycbcrCoeffs[2]
Packit 7838c8
Packit 7838c8
uint16	compression = COMPRESSION_PACKBITS;
Packit 7838c8
uint32	rowsperstrip = (uint32) -1;
Packit 7838c8
Packit 7838c8
uint16	horizSubSampling = 2;		/* YCbCr horizontal subsampling */
Packit 7838c8
uint16	vertSubSampling = 2;		/* YCbCr vertical subsampling */
Packit 7838c8
float	ycbcrCoeffs[3] = { .299F, .587F, .114F };
Packit 7838c8
/* default coding range is CCIR Rec 601-1 with no headroom/footroom */
Packit 7838c8
float	refBlackWhite[6] = { 0.F, 255.F, 128.F, 255.F, 128.F, 255.F };
Packit 7838c8
Packit 7838c8
static	int tiffcvt(TIFF* in, TIFF* out);
Packit 7838c8
static	void usage(int code);
Packit 7838c8
static	void setupLumaTables(void);
Packit 7838c8
Packit 7838c8
int
Packit 7838c8
main(int argc, char* argv[])
Packit 7838c8
{
Packit 7838c8
	TIFF *in, *out;
Packit 7838c8
	int c;
Packit 7838c8
#if !HAVE_DECL_OPTARG
Packit 7838c8
	extern int optind;
Packit 7838c8
	extern char *optarg;
Packit 7838c8
#endif
Packit 7838c8
Packit 7838c8
	while ((c = getopt(argc, argv, "c:h:r:v:z")) != -1)
Packit 7838c8
		switch (c) {
Packit 7838c8
		case 'c':
Packit 7838c8
			if (streq(optarg, "none"))
Packit 7838c8
			    compression = COMPRESSION_NONE;
Packit 7838c8
			else if (streq(optarg, "packbits"))
Packit 7838c8
			    compression = COMPRESSION_PACKBITS;
Packit 7838c8
			else if (streq(optarg, "lzw"))
Packit 7838c8
			    compression = COMPRESSION_LZW;
Packit 7838c8
			else if (streq(optarg, "jpeg"))
Packit 7838c8
			    compression = COMPRESSION_JPEG;
Packit 7838c8
			else if (streq(optarg, "zip"))
Packit 7838c8
			    compression = COMPRESSION_ADOBE_DEFLATE;
Packit 7838c8
			else
Packit 7838c8
			    usage(-1);
Packit 7838c8
			break;
Packit 7838c8
		case 'h':
Packit 7838c8
			horizSubSampling = atoi(optarg);
Packit 7838c8
            if( horizSubSampling != 1 && horizSubSampling != 2 && horizSubSampling != 4 )
Packit 7838c8
                usage(-1);
Packit 7838c8
			break;
Packit 7838c8
		case 'v':
Packit 7838c8
			vertSubSampling = atoi(optarg);
Packit 7838c8
            if( vertSubSampling != 1 && vertSubSampling != 2 && vertSubSampling != 4 )
Packit 7838c8
                usage(-1);
Packit 7838c8
			break;
Packit 7838c8
		case 'r':
Packit 7838c8
			rowsperstrip = atoi(optarg);
Packit 7838c8
			break;
Packit 7838c8
		case 'z':	/* CCIR Rec 601-1 w/ headroom/footroom */
Packit 7838c8
			refBlackWhite[0] = 16.;
Packit 7838c8
			refBlackWhite[1] = 235.;
Packit 7838c8
			refBlackWhite[2] = 128.;
Packit 7838c8
			refBlackWhite[3] = 240.;
Packit 7838c8
			refBlackWhite[4] = 128.;
Packit 7838c8
			refBlackWhite[5] = 240.;
Packit 7838c8
			break;
Packit 7838c8
		case '?':
Packit 7838c8
			usage(0);
Packit 7838c8
			/*NOTREACHED*/
Packit 7838c8
		}
Packit 7838c8
	if (argc - optind < 2)
Packit 7838c8
		usage(-1);
Packit 7838c8
	out = TIFFOpen(argv[argc-1], "w");
Packit 7838c8
	if (out == NULL)
Packit 7838c8
		return (-2);
Packit 7838c8
	setupLumaTables();
Packit 7838c8
	for (; optind < argc-1; optind++) {
Packit 7838c8
		in = TIFFOpen(argv[optind], "r");
Packit 7838c8
		if (in != NULL) {
Packit 7838c8
			do {
Packit 7838c8
				if (!tiffcvt(in, out) ||
Packit 7838c8
				    !TIFFWriteDirectory(out)) {
Packit 7838c8
					(void) TIFFClose(out);
Packit 7838c8
					return (1);
Packit 7838c8
				}
Packit 7838c8
			} while (TIFFReadDirectory(in));
Packit 7838c8
			(void) TIFFClose(in);
Packit 7838c8
		}
Packit 7838c8
	}
Packit 7838c8
	(void) TIFFClose(out);
Packit 7838c8
	return (0);
Packit 7838c8
}
Packit 7838c8
Packit 7838c8
float	*lumaRed;
Packit 7838c8
float	*lumaGreen;
Packit 7838c8
float	*lumaBlue;
Packit 7838c8
float	D1, D2;
Packit 7838c8
int	Yzero;
Packit 7838c8
Packit 7838c8
static float*
Packit 7838c8
setupLuma(float c)
Packit 7838c8
{
Packit 7838c8
	float *v = (float *)_TIFFmalloc(256 * sizeof (float));
Packit 7838c8
	int i;
Packit 7838c8
	for (i = 0; i < 256; i++)
Packit 7838c8
		v[i] = c * i;
Packit 7838c8
	return (v);
Packit 7838c8
}
Packit 7838c8
Packit 7838c8
static unsigned
Packit 7838c8
V2Code(float f, float RB, float RW, int CR)
Packit 7838c8
{
Packit 7838c8
	unsigned int c = (unsigned int)((((f)*(RW-RB)/CR)+RB)+.5);
Packit 7838c8
	return (c > 255 ? 255 : c);
Packit 7838c8
}
Packit 7838c8
Packit 7838c8
static void
Packit 7838c8
setupLumaTables(void)
Packit 7838c8
{
Packit 7838c8
	lumaRed = setupLuma(LumaRed);
Packit 7838c8
	lumaGreen = setupLuma(LumaGreen);
Packit 7838c8
	lumaBlue = setupLuma(LumaBlue);
Packit 7838c8
	D1 = 1.F/(2.F - 2.F*LumaBlue);
Packit 7838c8
	D2 = 1.F/(2.F - 2.F*LumaRed);
Packit 7838c8
	Yzero = V2Code(0, refBlackWhite[0], refBlackWhite[1], 255);
Packit 7838c8
}
Packit 7838c8
Packit 7838c8
static void
Packit 7838c8
cvtClump(unsigned char* op, uint32* raster, uint32 ch, uint32 cw, uint32 w)
Packit 7838c8
{
Packit 7838c8
	float Y, Cb = 0, Cr = 0;
Packit 7838c8
	uint32 j, k;
Packit 7838c8
	/*
Packit 7838c8
	 * Convert ch-by-cw block of RGB
Packit 7838c8
	 * to YCbCr and sample accordingly.
Packit 7838c8
	 */
Packit 7838c8
	for (k = 0; k < ch; k++) {
Packit 7838c8
		for (j = 0; j < cw; j++) {
Packit 7838c8
			uint32 RGB = (raster - k*w)[j];
Packit 7838c8
			Y = lumaRed[TIFFGetR(RGB)] +
Packit 7838c8
			    lumaGreen[TIFFGetG(RGB)] +
Packit 7838c8
			    lumaBlue[TIFFGetB(RGB)];
Packit 7838c8
			/* accumulate chrominance */
Packit 7838c8
			Cb += (TIFFGetB(RGB) - Y) * D1;
Packit 7838c8
			Cr += (TIFFGetR(RGB) - Y) * D2;
Packit 7838c8
			/* emit luminence */
Packit 7838c8
			*op++ = V2Code(Y,
Packit 7838c8
			    refBlackWhite[0], refBlackWhite[1], 255);
Packit 7838c8
		}
Packit 7838c8
		for (; j < horizSubSampling; j++)
Packit 7838c8
			*op++ = Yzero;
Packit 7838c8
	}
Packit 7838c8
	for (; k < vertSubSampling; k++) {
Packit 7838c8
		for (j = 0; j < horizSubSampling; j++)
Packit 7838c8
			*op++ = Yzero;
Packit 7838c8
	}
Packit 7838c8
	/* emit sampled chrominance values */
Packit 7838c8
	*op++ = V2Code(Cb / (ch*cw), refBlackWhite[2], refBlackWhite[3], 127);
Packit 7838c8
	*op++ = V2Code(Cr / (ch*cw), refBlackWhite[4], refBlackWhite[5], 127);
Packit 7838c8
}
Packit 7838c8
#undef LumaRed
Packit 7838c8
#undef LumaGreen
Packit 7838c8
#undef LumaBlue
Packit 7838c8
#undef V2Code
Packit 7838c8
Packit 7838c8
/*
Packit 7838c8
 * Convert a strip of RGB data to YCbCr and
Packit 7838c8
 * sample to generate the output data.
Packit 7838c8
 */
Packit 7838c8
static void
Packit 7838c8
cvtStrip(unsigned char* op, uint32* raster, uint32 nrows, uint32 width)
Packit 7838c8
{
Packit 7838c8
	uint32 x;
Packit 7838c8
	int clumpSize = vertSubSampling * horizSubSampling + 2;
Packit 7838c8
	uint32 *tp;
Packit 7838c8
Packit 7838c8
	for (; nrows >= vertSubSampling; nrows -= vertSubSampling) {
Packit 7838c8
		tp = raster;
Packit 7838c8
		for (x = width; x >= horizSubSampling; x -= horizSubSampling) {
Packit 7838c8
			cvtClump(op, tp,
Packit 7838c8
			    vertSubSampling, horizSubSampling, width);
Packit 7838c8
			op += clumpSize;
Packit 7838c8
			tp += horizSubSampling;
Packit 7838c8
		}
Packit 7838c8
		if (x > 0) {
Packit 7838c8
			cvtClump(op, tp, vertSubSampling, x, width);
Packit 7838c8
			op += clumpSize;
Packit 7838c8
		}
Packit 7838c8
		raster -= vertSubSampling*width;
Packit 7838c8
	}
Packit 7838c8
	if (nrows > 0) {
Packit 7838c8
		tp = raster;
Packit 7838c8
		for (x = width; x >= horizSubSampling; x -= horizSubSampling) {
Packit 7838c8
			cvtClump(op, tp, nrows, horizSubSampling, width);
Packit 7838c8
			op += clumpSize;
Packit 7838c8
			tp += horizSubSampling;
Packit 7838c8
		}
Packit 7838c8
		if (x > 0)
Packit 7838c8
			cvtClump(op, tp, nrows, x, width);
Packit 7838c8
	}
Packit 7838c8
}
Packit 7838c8
Packit 7838c8
static int
Packit 7838c8
cvtRaster(TIFF* tif, uint32* raster, uint32 width, uint32 height)
Packit 7838c8
{
Packit 7838c8
	uint32 y;
Packit 7838c8
	tstrip_t strip = 0;
Packit 7838c8
	tsize_t cc, acc;
Packit 7838c8
	unsigned char* buf;
Packit 7838c8
	uint32 rwidth = roundup(width, horizSubSampling);
Packit 7838c8
	uint32 rheight = roundup(height, vertSubSampling);
Packit 7838c8
	uint32 nrows = (rowsperstrip > rheight ? rheight : rowsperstrip);
Packit 7838c8
        uint32 rnrows = roundup(nrows,vertSubSampling);
Packit 7838c8
Packit 7838c8
	cc = rnrows*rwidth +
Packit 7838c8
	    2*((rnrows*rwidth) / (horizSubSampling*vertSubSampling));
Packit 7838c8
	buf = (unsigned char*)_TIFFmalloc(cc);
Packit 7838c8
	// FIXME unchecked malloc
Packit 7838c8
	for (y = height; (int32) y > 0; y -= nrows) {
Packit 7838c8
		uint32 nr = (y > nrows ? nrows : y);
Packit 7838c8
		cvtStrip(buf, raster + (y-1)*width, nr, width);
Packit 7838c8
		nr = roundup(nr, vertSubSampling);
Packit 7838c8
		acc = nr*rwidth +
Packit 7838c8
			2*((nr*rwidth)/(horizSubSampling*vertSubSampling));
Packit 7838c8
		if (!TIFFWriteEncodedStrip(tif, strip++, buf, acc)) {
Packit 7838c8
			_TIFFfree(buf);
Packit 7838c8
			return (0);
Packit 7838c8
		}
Packit 7838c8
	}
Packit 7838c8
	_TIFFfree(buf);
Packit 7838c8
	return (1);
Packit 7838c8
}
Packit 7838c8
Packit 7838c8
static int
Packit 7838c8
tiffcvt(TIFF* in, TIFF* out)
Packit 7838c8
{
Packit 7838c8
	uint32 width, height;		/* image width & height */
Packit 7838c8
	uint32* raster;			/* retrieve RGBA image */
Packit 7838c8
	uint16 shortv;
Packit 7838c8
	float floatv;
Packit 7838c8
	char *stringv;
Packit 7838c8
	uint32 longv;
Packit 7838c8
	int result;
Packit 7838c8
	size_t pixel_count;
Packit 7838c8
Packit 7838c8
	TIFFGetField(in, TIFFTAG_IMAGEWIDTH, &width);
Packit 7838c8
	TIFFGetField(in, TIFFTAG_IMAGELENGTH, &height);
Packit 7838c8
	pixel_count = width * height;
Packit 7838c8
Packit 7838c8
 	/* XXX: Check the integer overflow. */
Packit 7838c8
 	if (!width || !height || pixel_count / width != height) {
Packit 7838c8
 		TIFFError(TIFFFileName(in),
Packit 7838c8
 			  "Malformed input file; "
Packit 7838c8
 			  "can't allocate buffer for raster of %lux%lu size",
Packit 7838c8
 			  (unsigned long)width, (unsigned long)height);
Packit 7838c8
 		return 0;
Packit 7838c8
 	}
Packit 7838c8
 
Packit 7838c8
 	raster = (uint32*)_TIFFCheckMalloc(in, pixel_count, sizeof(uint32),
Packit 7838c8
 					   "raster buffer");
Packit 7838c8
  	if (raster == 0) {
Packit 7838c8
 		TIFFError(TIFFFileName(in),
Packit 7838c8
 			  "Failed to allocate buffer (%lu elements of %lu each)",
Packit 7838c8
 			  (unsigned long)pixel_count,
Packit 7838c8
 			  (unsigned long)sizeof(uint32));
Packit 7838c8
  		return (0);
Packit 7838c8
  	}
Packit 7838c8
Packit 7838c8
	if (!TIFFReadRGBAImage(in, width, height, raster, 0)) {
Packit 7838c8
		_TIFFfree(raster);
Packit 7838c8
		return (0);
Packit 7838c8
	}
Packit 7838c8
Packit 7838c8
	CopyField(TIFFTAG_SUBFILETYPE, longv);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_IMAGEWIDTH, width);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_IMAGELENGTH, height);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_BITSPERSAMPLE, 8);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_COMPRESSION, compression);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_YCBCR);
Packit 7838c8
	if (compression == COMPRESSION_JPEG)
Packit 7838c8
		TIFFSetField(out, TIFFTAG_JPEGCOLORMODE, JPEGCOLORMODE_RAW);
Packit 7838c8
	CopyField(TIFFTAG_FILLORDER, shortv);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_ORIENTATION, ORIENTATION_TOPLEFT);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_SAMPLESPERPIXEL, 3);
Packit 7838c8
	CopyField(TIFFTAG_XRESOLUTION, floatv);
Packit 7838c8
	CopyField(TIFFTAG_YRESOLUTION, floatv);
Packit 7838c8
	CopyField(TIFFTAG_RESOLUTIONUNIT, shortv);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
Packit 7838c8
	{ char buf[2048];
Packit 7838c8
	  char *cp = strrchr(TIFFFileName(in), '/');
Packit 7838c8
	  snprintf(buf, sizeof(buf), "YCbCr conversion of %s",
Packit 7838c8
		   cp ? cp+1 : TIFFFileName(in));
Packit 7838c8
	  TIFFSetField(out, TIFFTAG_IMAGEDESCRIPTION, buf);
Packit 7838c8
	}
Packit 7838c8
	TIFFSetField(out, TIFFTAG_SOFTWARE, TIFFGetVersion());
Packit 7838c8
	CopyField(TIFFTAG_DOCUMENTNAME, stringv);
Packit 7838c8
Packit 7838c8
	TIFFSetField(out, TIFFTAG_REFERENCEBLACKWHITE, refBlackWhite);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_YCBCRSUBSAMPLING,
Packit 7838c8
	    horizSubSampling, vertSubSampling);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_YCBCRPOSITIONING, YCBCRPOSITION_CENTERED);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_YCBCRCOEFFICIENTS, ycbcrCoeffs);
Packit 7838c8
	rowsperstrip = TIFFDefaultStripSize(out, rowsperstrip);
Packit 7838c8
	TIFFSetField(out, TIFFTAG_ROWSPERSTRIP, rowsperstrip);
Packit 7838c8
Packit 7838c8
	result = cvtRaster(out, raster, width, height);
Packit 7838c8
        _TIFFfree(raster);
Packit 7838c8
        return result;
Packit 7838c8
}
Packit 7838c8
Packit 7838c8
char* stuff[] = {
Packit 7838c8
    "usage: rgb2ycbcr [-c comp] [-r rows] [-h N] [-v N] input... output\n",
Packit 7838c8
    "where comp is one of the following compression algorithms:\n",
Packit 7838c8
    " jpeg\t\tJPEG encoding\n",
Packit 7838c8
    " lzw\t\tLempel-Ziv & Welch encoding\n",
Packit 7838c8
    " zip\t\tdeflate encoding\n",
Packit 7838c8
    " packbits\tPackBits encoding (default)\n",
Packit 7838c8
    " none\t\tno compression\n",
Packit 7838c8
    "and the other options are:\n",
Packit 7838c8
    " -r\trows/strip\n",
Packit 7838c8
    " -h\thorizontal sampling factor (1,2,4)\n",
Packit 7838c8
    " -v\tvertical sampling factor (1,2,4)\n",
Packit 7838c8
    NULL
Packit 7838c8
};
Packit 7838c8
Packit 7838c8
static void
Packit 7838c8
usage(int code)
Packit 7838c8
{
Packit 7838c8
	char buf[BUFSIZ];
Packit 7838c8
	int i;
Packit 7838c8
Packit 7838c8
	setbuf(stderr, buf);
Packit 7838c8
       
Packit 7838c8
 fprintf(stderr, "%s\n\n", TIFFGetVersion());
Packit 7838c8
	for (i = 0; stuff[i] != NULL; i++)
Packit 7838c8
		fprintf(stderr, "%s\n", stuff[i]);
Packit 7838c8
	exit(code);
Packit 7838c8
}
Packit 7838c8
Packit 7838c8
/* vim: set ts=8 sts=8 sw=8 noet: */
Packit 7838c8
/*
Packit 7838c8
 * Local Variables:
Packit 7838c8
 * mode: c
Packit 7838c8
 * c-basic-offset: 8
Packit 7838c8
 * fill-column: 78
Packit 7838c8
 * End:
Packit 7838c8
 */