461 lines
15 KiB
C
461 lines
15 KiB
C
/*
|
|
* MSCMS - Color Management System for Wine
|
|
*
|
|
* Copyright 2005, 2006, 2008 Hans Leidekker
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "wine/debug.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winnls.h"
|
|
#include "wingdi.h"
|
|
#include "winuser.h"
|
|
#include "icm.h"
|
|
|
|
#include "mscms_priv.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(mscms);
|
|
|
|
#ifdef HAVE_LCMS
|
|
|
|
static DWORD from_profile( HPROFILE profile )
|
|
{
|
|
PROFILEHEADER header;
|
|
|
|
GetColorProfileHeader( profile, &header );
|
|
TRACE( "color space: 0x%08x %s\n", header.phDataColorSpace, MSCMS_dbgstr_tag( header.phDataColorSpace ) );
|
|
|
|
switch (header.phDataColorSpace)
|
|
{
|
|
case 0x434d594b: return TYPE_CMYK_16; /* 'CMYK' */
|
|
case 0x47524159: return TYPE_GRAY_16; /* 'GRAY' */
|
|
case 0x4c616220: return TYPE_Lab_16; /* 'Lab ' */
|
|
case 0x52474220: return TYPE_RGB_16; /* 'RGB ' */
|
|
case 0x58595a20: return TYPE_XYZ_16; /* 'XYZ ' */
|
|
default:
|
|
WARN("unhandled format\n");
|
|
return TYPE_RGB_16;
|
|
}
|
|
}
|
|
|
|
static DWORD from_bmformat( BMFORMAT format )
|
|
{
|
|
static int quietfixme = 0;
|
|
TRACE( "bitmap format: 0x%08x\n", format );
|
|
|
|
switch (format)
|
|
{
|
|
case BM_RGBTRIPLETS: return TYPE_RGB_8;
|
|
case BM_BGRTRIPLETS: return TYPE_BGR_8;
|
|
case BM_GRAY: return TYPE_GRAY_8;
|
|
default:
|
|
if (quietfixme == 0)
|
|
{
|
|
FIXME("unhandled bitmap format 0x%x\n", format);
|
|
quietfixme = 1;
|
|
}
|
|
return TYPE_RGB_8;
|
|
}
|
|
}
|
|
|
|
static DWORD from_type( COLORTYPE type )
|
|
{
|
|
TRACE( "color type: 0x%08x\n", type );
|
|
|
|
switch (type)
|
|
{
|
|
case COLOR_GRAY: return TYPE_GRAY_16;
|
|
case COLOR_RGB: return TYPE_RGB_16;
|
|
case COLOR_XYZ: return TYPE_XYZ_16;
|
|
case COLOR_Yxy: return TYPE_Yxy_16;
|
|
case COLOR_Lab: return TYPE_Lab_16;
|
|
case COLOR_CMYK: return TYPE_CMYK_16;
|
|
default:
|
|
FIXME("unhandled color type\n");
|
|
return TYPE_RGB_16;
|
|
}
|
|
}
|
|
|
|
#endif /* HAVE_LCMS */
|
|
|
|
/******************************************************************************
|
|
* CreateColorTransformA [MSCMS.@]
|
|
*
|
|
* See CreateColorTransformW.
|
|
*/
|
|
HTRANSFORM WINAPI CreateColorTransformA( LPLOGCOLORSPACEA space, HPROFILE dest,
|
|
HPROFILE target, DWORD flags )
|
|
{
|
|
LOGCOLORSPACEW spaceW;
|
|
DWORD len;
|
|
|
|
TRACE( "( %p, %p, %p, 0x%08x )\n", space, dest, target, flags );
|
|
|
|
if (!space || !dest) return FALSE;
|
|
|
|
memcpy( &spaceW, space, FIELD_OFFSET(LOGCOLORSPACEA, lcsFilename) );
|
|
spaceW.lcsSize = sizeof(LOGCOLORSPACEW);
|
|
|
|
len = MultiByteToWideChar( CP_ACP, 0, space->lcsFilename, -1, NULL, 0 );
|
|
MultiByteToWideChar( CP_ACP, 0, space->lcsFilename, -1, spaceW.lcsFilename, len );
|
|
|
|
return CreateColorTransformW( &spaceW, dest, target, flags );
|
|
}
|
|
|
|
/******************************************************************************
|
|
* CreateColorTransformW [MSCMS.@]
|
|
*
|
|
* Create a color transform.
|
|
*
|
|
* PARAMS
|
|
* space [I] Input color space.
|
|
* dest [I] Color profile of destination device.
|
|
* target [I] Color profile of target device.
|
|
* flags [I] Flags.
|
|
*
|
|
* RETURNS
|
|
* Success: Handle to a transform.
|
|
* Failure: NULL
|
|
*/
|
|
HTRANSFORM WINAPI CreateColorTransformW( LPLOGCOLORSPACEW space, HPROFILE dest,
|
|
HPROFILE target, DWORD flags )
|
|
{
|
|
HTRANSFORM ret = NULL;
|
|
#ifdef HAVE_LCMS
|
|
struct transform transform;
|
|
struct profile *dst, *tgt = NULL;
|
|
cmsHPROFILE cmsinput, cmsoutput, cmstarget = NULL;
|
|
DWORD in_format, out_format, proofing = 0;
|
|
int intent;
|
|
|
|
TRACE( "( %p, %p, %p, 0x%08x )\n", space, dest, target, flags );
|
|
|
|
if (!space || !(dst = grab_profile( dest ))) return FALSE;
|
|
|
|
if (target && !(tgt = grab_profile( target )))
|
|
{
|
|
release_profile( dst );
|
|
return FALSE;
|
|
}
|
|
intent = space->lcsIntent > 3 ? INTENT_PERCEPTUAL : space->lcsIntent;
|
|
|
|
TRACE( "lcsIntent: %x\n", space->lcsIntent );
|
|
TRACE( "lcsCSType: %s\n", MSCMS_dbgstr_tag( space->lcsCSType ) );
|
|
TRACE( "lcsFilename: %s\n", debugstr_w( space->lcsFilename ) );
|
|
|
|
in_format = TYPE_RGB_16;
|
|
out_format = from_profile( dest );
|
|
|
|
cmsinput = cmsCreate_sRGBProfile(); /* FIXME: create from supplied color space */
|
|
if (target)
|
|
{
|
|
proofing = cmsFLAGS_SOFTPROOFING;
|
|
cmstarget = tgt->cmsprofile;
|
|
}
|
|
cmsoutput = dst->cmsprofile;
|
|
transform.cmstransform = cmsCreateProofingTransform(cmsinput, in_format, cmsoutput, out_format, cmstarget,
|
|
intent, INTENT_ABSOLUTE_COLORIMETRIC, proofing);
|
|
|
|
ret = create_transform( &transform );
|
|
|
|
if (tgt) release_profile( tgt );
|
|
release_profile( dst );
|
|
|
|
#endif /* HAVE_LCMS */
|
|
return ret;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* CreateMultiProfileTransform [MSCMS.@]
|
|
*
|
|
* Create a color transform from an array of color profiles.
|
|
*
|
|
* PARAMS
|
|
* profiles [I] Array of color profiles.
|
|
* nprofiles [I] Number of color profiles.
|
|
* intents [I] Array of rendering intents.
|
|
* flags [I] Flags.
|
|
* cmm [I] Profile to take the CMM from.
|
|
*
|
|
* RETURNS
|
|
* Success: Handle to a transform.
|
|
* Failure: NULL
|
|
*/
|
|
HTRANSFORM WINAPI CreateMultiProfileTransform( PHPROFILE profiles, DWORD nprofiles,
|
|
PDWORD intents, DWORD nintents, DWORD flags, DWORD cmm )
|
|
{
|
|
HTRANSFORM ret = NULL;
|
|
#ifdef HAVE_LCMS
|
|
cmsHPROFILE *cmsprofiles, cmsconvert = NULL;
|
|
struct transform transform;
|
|
struct profile *profile0, *profile1;
|
|
DWORD in_format, out_format;
|
|
|
|
TRACE( "( %p, 0x%08x, %p, 0x%08x, 0x%08x, 0x%08x )\n",
|
|
profiles, nprofiles, intents, nintents, flags, cmm );
|
|
|
|
if (!profiles || !nprofiles || !intents) return NULL;
|
|
|
|
if (nprofiles > 2)
|
|
{
|
|
FIXME("more than 2 profiles not supported\n");
|
|
return NULL;
|
|
}
|
|
|
|
profile0 = grab_profile( profiles[0] );
|
|
if (!profile0) return NULL;
|
|
profile1 = grab_profile( profiles[1] );
|
|
if (!profile1)
|
|
{
|
|
release_profile( profile0 );
|
|
return NULL;
|
|
}
|
|
in_format = from_profile( profiles[0] );
|
|
out_format = from_profile( profiles[nprofiles - 1] );
|
|
|
|
if (in_format != out_format)
|
|
{
|
|
/* insert a conversion profile for pairings that lcms doesn't handle */
|
|
if (out_format == TYPE_RGB_16) cmsconvert = cmsCreate_sRGBProfile();
|
|
if (out_format == TYPE_Lab_16) cmsconvert = cmsCreateLabProfile( NULL );
|
|
}
|
|
|
|
cmsprofiles = HeapAlloc( GetProcessHeap(), 0, (nprofiles + 1) * sizeof(cmsHPROFILE *) );
|
|
if (cmsprofiles)
|
|
{
|
|
cmsprofiles[0] = profile0->cmsprofile;
|
|
if (cmsconvert)
|
|
{
|
|
cmsprofiles[1] = cmsconvert;
|
|
cmsprofiles[2] = profile1->cmsprofile;
|
|
nprofiles++;
|
|
}
|
|
else
|
|
{
|
|
cmsprofiles[1] = profile1->cmsprofile;
|
|
}
|
|
transform.cmstransform = cmsCreateMultiprofileTransform( cmsprofiles, nprofiles, in_format, out_format, *intents, 0 );
|
|
|
|
HeapFree( GetProcessHeap(), 0, cmsprofiles );
|
|
ret = create_transform( &transform );
|
|
}
|
|
|
|
release_profile( profile0 );
|
|
release_profile( profile1 );
|
|
|
|
#endif /* HAVE_LCMS */
|
|
return ret;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* DeleteColorTransform [MSCMS.@]
|
|
*
|
|
* Delete a color transform.
|
|
*
|
|
* PARAMS
|
|
* transform [I] Handle to a color transform.
|
|
*
|
|
* RETURNS
|
|
* Success: TRUE
|
|
* Failure: FALSE
|
|
*/
|
|
BOOL WINAPI DeleteColorTransform( HTRANSFORM handle )
|
|
{
|
|
BOOL ret = FALSE;
|
|
#ifdef HAVE_LCMS
|
|
|
|
TRACE( "( %p )\n", handle );
|
|
|
|
ret = close_transform( handle );
|
|
|
|
#endif /* HAVE_LCMS */
|
|
return ret;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* TranslateBitmapBits [MSCMS.@]
|
|
*
|
|
* Perform color translation.
|
|
*
|
|
* PARAMS
|
|
* transform [I] Handle to a color transform.
|
|
* srcbits [I] Source bitmap.
|
|
* input [I] Format of the source bitmap.
|
|
* width [I] Width of the source bitmap.
|
|
* height [I] Height of the source bitmap.
|
|
* inputstride [I] Number of bytes in one scanline.
|
|
* destbits [I] Destination bitmap.
|
|
* output [I] Format of the destination bitmap.
|
|
* outputstride [I] Number of bytes in one scanline.
|
|
* callback [I] Callback function.
|
|
* data [I] Callback data.
|
|
*
|
|
* RETURNS
|
|
* Success: TRUE
|
|
* Failure: FALSE
|
|
*/
|
|
BOOL WINAPI TranslateBitmapBits( HTRANSFORM handle, PVOID srcbits, BMFORMAT input,
|
|
DWORD width, DWORD height, DWORD inputstride, PVOID destbits, BMFORMAT output,
|
|
DWORD outputstride, PBMCALLBACKFN callback, ULONG data )
|
|
{
|
|
BOOL ret = FALSE;
|
|
#ifdef HAVE_LCMS
|
|
struct transform *transform = grab_transform( handle );
|
|
|
|
TRACE( "( %p, %p, 0x%08x, 0x%08x, 0x%08x, 0x%08x, %p, 0x%08x, 0x%08x, %p, 0x%08x )\n",
|
|
handle, srcbits, input, width, height, inputstride, destbits, output,
|
|
outputstride, callback, data );
|
|
|
|
if (!transform) return FALSE;
|
|
cmsChangeBuffersFormat( transform->cmstransform, from_bmformat(input), from_bmformat(output) );
|
|
|
|
cmsDoTransform( transform->cmstransform, srcbits, destbits, width * height );
|
|
release_transform( transform );
|
|
ret = TRUE;
|
|
|
|
#endif /* HAVE_LCMS */
|
|
return ret;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* TranslateColors [MSCMS.@]
|
|
*
|
|
* Perform color translation.
|
|
*
|
|
* PARAMS
|
|
* transform [I] Handle to a color transform.
|
|
* input [I] Array of input colors.
|
|
* number [I] Number of colors to translate.
|
|
* input_type [I] Input color format.
|
|
* output [O] Array of output colors.
|
|
* output_type [I] Output color format.
|
|
*
|
|
* RETURNS
|
|
* Success: TRUE
|
|
* Failure: FALSE
|
|
*/
|
|
BOOL WINAPI TranslateColors( HTRANSFORM handle, PCOLOR in, DWORD count,
|
|
COLORTYPE input_type, PCOLOR out, COLORTYPE output_type )
|
|
{
|
|
#ifdef HAVE_LCMS
|
|
BOOL ret = TRUE;
|
|
struct transform *transform = grab_transform( handle );
|
|
cmsHTRANSFORM xfrm;
|
|
unsigned int i;
|
|
|
|
TRACE( "( %p, %p, %d, %d, %p, %d )\n", handle, in, count, input_type, out, output_type );
|
|
|
|
if (!transform) return FALSE;
|
|
|
|
xfrm = transform->cmstransform;
|
|
cmsChangeBuffersFormat( xfrm, from_type(input_type), from_type(output_type) );
|
|
|
|
switch (input_type)
|
|
{
|
|
case COLOR_RGB:
|
|
{
|
|
switch (output_type)
|
|
{
|
|
case COLOR_RGB: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].rgb, &out[i].rgb, 1 ); goto done;
|
|
case COLOR_Lab: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].rgb, &out[i].Lab, 1 ); goto done;
|
|
case COLOR_GRAY: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].rgb, &out[i].gray, 1 ); goto done;
|
|
case COLOR_CMYK: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].rgb, &out[i].cmyk, 1 ); goto done;
|
|
case COLOR_XYZ: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].rgb, &out[i].XYZ, 1 ); goto done;
|
|
default:
|
|
FIXME("unhandled input/output pair: %d/%d\n", input_type, output_type);
|
|
ret = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
case COLOR_Lab:
|
|
{
|
|
switch (output_type)
|
|
{
|
|
case COLOR_RGB: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].Lab, &out[i].rgb, 1 ); goto done;
|
|
case COLOR_Lab: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].Lab, &out[i].Lab, 1 ); goto done;
|
|
case COLOR_GRAY: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].Lab, &out[i].gray, 1 ); goto done;
|
|
case COLOR_CMYK: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].Lab, &out[i].cmyk, 1 ); goto done;
|
|
case COLOR_XYZ: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].Lab, &out[i].XYZ, 1 ); goto done;
|
|
default:
|
|
FIXME("unhandled input/output pair: %d/%d\n", input_type, output_type);
|
|
ret = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
case COLOR_GRAY:
|
|
{
|
|
switch (output_type)
|
|
{
|
|
case COLOR_RGB: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].gray, &out[i].rgb, 1 ); goto done;
|
|
case COLOR_Lab: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].gray, &out[i].Lab, 1 ); goto done;
|
|
case COLOR_GRAY: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].gray, &out[i].gray, 1 ); goto done;
|
|
case COLOR_CMYK: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].gray, &out[i].cmyk, 1 ); goto done;
|
|
case COLOR_XYZ: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].gray, &out[i].XYZ, 1 ); goto done;
|
|
default:
|
|
FIXME("unhandled input/output pair: %d/%d\n", input_type, output_type);
|
|
ret = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
case COLOR_CMYK:
|
|
{
|
|
switch (output_type)
|
|
{
|
|
case COLOR_RGB: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].cmyk, &out[i].rgb, 1 ); goto done;
|
|
case COLOR_Lab: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].cmyk, &out[i].Lab, 1 ); goto done;
|
|
case COLOR_GRAY: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].cmyk, &out[i].gray, 1 ); goto done;
|
|
case COLOR_CMYK: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].cmyk, &out[i].cmyk, 1 ); goto done;
|
|
case COLOR_XYZ: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].cmyk, &out[i].XYZ, 1 ); goto done;
|
|
default:
|
|
FIXME("unhandled input/output pair: %d/%d\n", input_type, output_type);
|
|
ret = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
case COLOR_XYZ:
|
|
{
|
|
switch (output_type)
|
|
{
|
|
case COLOR_RGB: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].XYZ, &out[i].rgb, 1 ); goto done;
|
|
case COLOR_Lab: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].XYZ, &out[i].Lab, 1 ); goto done;
|
|
case COLOR_GRAY: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].XYZ, &out[i].gray, 1 ); goto done;
|
|
case COLOR_CMYK: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].XYZ, &out[i].cmyk, 1 ); goto done;
|
|
case COLOR_XYZ: for (i = 0; i < count; i++) cmsDoTransform( xfrm, &in[i].XYZ, &out[i].XYZ, 1 ); goto done;
|
|
default:
|
|
FIXME("unhandled input/output pair: %d/%d\n", input_type, output_type);
|
|
ret = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
default:
|
|
FIXME("unhandled input/output pair: %d/%d\n", input_type, output_type);
|
|
ret = FALSE;
|
|
break;
|
|
}
|
|
|
|
done:
|
|
release_transform( transform );
|
|
return ret;
|
|
|
|
#endif /* HAVE_LCMS */
|
|
return FALSE;
|
|
}
|