| /* |
| * 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_LCMS2 |
| |
| static DWORD from_profile( HPROFILE profile ) |
| { |
| PROFILEHEADER header; |
| |
| GetColorProfileHeader( profile, &header ); |
| TRACE( "color space: 0x%08x %s\n", header.phDataColorSpace, 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 BOOL quietfixme = FALSE; |
| 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; |
| case BM_xRGBQUADS: return TYPE_ARGB_8; |
| case BM_xBGRQUADS: return TYPE_ABGR_8; |
| default: |
| if (!quietfixme) |
| { |
| FIXME("unhandled bitmap format 0x%x\n", format); |
| quietfixme = TRUE; |
| } |
| 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_LCMS2 */ |
| |
| /****************************************************************************** |
| * 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_LCMS2 |
| 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", 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_LCMS2 */ |
| 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_LCMS2 |
| 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 = cmsCreateLab2Profile( 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_LCMS2 */ |
| 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_LCMS2 |
| |
| TRACE( "( %p )\n", handle ); |
| |
| ret = close_transform( handle ); |
| |
| #endif /* HAVE_LCMS2 */ |
| 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_LCMS2 |
| 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_LCMS2 */ |
| 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_LCMS2 |
| 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; |
| } |
| 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; |
| } |
| 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; |
| } |
| 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; |
| } |
| 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; |
| } |
| break; |
| } |
| default: |
| FIXME("unhandled input/output pair: %d/%d\n", input_type, output_type); |
| ret = FALSE; |
| break; |
| } |
| |
| done: |
| release_transform( transform ); |
| return ret; |
| |
| #else /* HAVE_LCMS2 */ |
| return FALSE; |
| #endif /* HAVE_LCMS2 */ |
| } |