2 * MSCMS - Color Management System for Wine
4 * Copyright 2005, 2006, 2008 Hans Leidekker
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #include "wine/debug.h"
33 #include "mscms_priv.h"
35 WINE_DEFAULT_DEBUG_CHANNEL(mscms
);
39 static DWORD
from_bmformat( BMFORMAT format
)
41 static BOOL quietfixme
= FALSE
;
46 case BM_RGBTRIPLETS
: ret
= TYPE_RGB_8
; break;
47 case BM_BGRTRIPLETS
: ret
= TYPE_BGR_8
; break;
48 case BM_GRAY
: ret
= TYPE_GRAY_8
; break;
49 case BM_xRGBQUADS
: ret
= TYPE_ARGB_8
; break;
50 case BM_xBGRQUADS
: ret
= TYPE_ABGR_8
; break;
51 case BM_KYMCQUADS
: ret
= TYPE_KYMC_8
; break;
55 FIXME( "unhandled bitmap format %08x\n", format
);
61 TRACE( "color space: %08x -> %08x\n", format
, ret
);
65 static DWORD
from_type( COLORTYPE type
)
71 case COLOR_GRAY
: ret
= TYPE_GRAY_16
; break;
72 case COLOR_RGB
: ret
= TYPE_RGB_16
; break;
73 case COLOR_XYZ
: ret
= TYPE_XYZ_16
; break;
74 case COLOR_Yxy
: ret
= TYPE_Yxy_16
; break;
75 case COLOR_Lab
: ret
= TYPE_Lab_16
; break;
76 case COLOR_CMYK
: ret
= TYPE_CMYK_16
; break;
78 FIXME( "unhandled color type %08x\n", type
);
83 TRACE( "color type: %08x -> %08x\n", type
, ret
);
87 #endif /* HAVE_LCMS2 */
89 /******************************************************************************
90 * CreateColorTransformA [MSCMS.@]
92 * See CreateColorTransformW.
94 HTRANSFORM WINAPI
CreateColorTransformA( LPLOGCOLORSPACEA space
, HPROFILE dest
,
95 HPROFILE target
, DWORD flags
)
97 LOGCOLORSPACEW spaceW
;
100 TRACE( "( %p, %p, %p, 0x%08x )\n", space
, dest
, target
, flags
);
102 if (!space
|| !dest
) return FALSE
;
104 memcpy( &spaceW
, space
, FIELD_OFFSET(LOGCOLORSPACEA
, lcsFilename
) );
105 spaceW
.lcsSize
= sizeof(LOGCOLORSPACEW
);
107 len
= MultiByteToWideChar( CP_ACP
, 0, space
->lcsFilename
, -1, NULL
, 0 );
108 MultiByteToWideChar( CP_ACP
, 0, space
->lcsFilename
, -1, spaceW
.lcsFilename
, len
);
110 return CreateColorTransformW( &spaceW
, dest
, target
, flags
);
113 /******************************************************************************
114 * CreateColorTransformW [MSCMS.@]
116 * Create a color transform.
119 * space [I] Input color space.
120 * dest [I] Color profile of destination device.
121 * target [I] Color profile of target device.
125 * Success: Handle to a transform.
128 HTRANSFORM WINAPI
CreateColorTransformW( LPLOGCOLORSPACEW space
, HPROFILE dest
,
129 HPROFILE target
, DWORD flags
)
131 HTRANSFORM ret
= NULL
;
133 struct transform transform
;
134 struct profile
*dst
, *tgt
= NULL
;
135 cmsHPROFILE cmsinput
, cmsoutput
, cmstarget
= NULL
;
139 TRACE( "( %p, %p, %p, 0x%08x )\n", space
, dest
, target
, flags
);
141 if (!space
|| !(dst
= grab_profile( dest
))) return FALSE
;
143 if (target
&& !(tgt
= grab_profile( target
)))
145 release_profile( dst
);
148 intent
= space
->lcsIntent
> 3 ? INTENT_PERCEPTUAL
: space
->lcsIntent
;
150 TRACE( "lcsIntent: %x\n", space
->lcsIntent
);
151 TRACE( "lcsCSType: %s\n", dbgstr_tag( space
->lcsCSType
) );
152 TRACE( "lcsFilename: %s\n", debugstr_w( space
->lcsFilename
) );
154 cmsinput
= cmsCreate_sRGBProfile(); /* FIXME: create from supplied color space */
157 proofing
= cmsFLAGS_SOFTPROOFING
;
158 cmstarget
= tgt
->cmsprofile
;
160 cmsoutput
= dst
->cmsprofile
;
161 transform
.cmstransform
= cmsCreateProofingTransform(cmsinput
, 0, cmsoutput
, 0, cmstarget
,
162 intent
, INTENT_ABSOLUTE_COLORIMETRIC
,
164 if (!transform
.cmstransform
)
166 if (tgt
) release_profile( tgt
);
167 release_profile( dst
);
171 ret
= create_transform( &transform
);
173 if (tgt
) release_profile( tgt
);
174 release_profile( dst
);
176 #endif /* HAVE_LCMS2 */
180 /******************************************************************************
181 * CreateMultiProfileTransform [MSCMS.@]
183 * Create a color transform from an array of color profiles.
186 * profiles [I] Array of color profiles.
187 * nprofiles [I] Number of color profiles.
188 * intents [I] Array of rendering intents.
190 * cmm [I] Profile to take the CMM from.
193 * Success: Handle to a transform.
196 HTRANSFORM WINAPI
CreateMultiProfileTransform( PHPROFILE profiles
, DWORD nprofiles
,
197 PDWORD intents
, DWORD nintents
, DWORD flags
, DWORD cmm
)
199 HTRANSFORM ret
= NULL
;
201 cmsHPROFILE
*cmsprofiles
;
202 struct transform transform
;
203 struct profile
*profile0
, *profile1
;
205 TRACE( "( %p, 0x%08x, %p, 0x%08x, 0x%08x, 0x%08x )\n",
206 profiles
, nprofiles
, intents
, nintents
, flags
, cmm
);
208 if (!profiles
|| !nprofiles
|| !intents
) return NULL
;
212 FIXME("more than 2 profiles not supported\n");
216 profile0
= grab_profile( profiles
[0] );
217 if (!profile0
) return NULL
;
218 profile1
= grab_profile( profiles
[1] );
221 release_profile( profile0
);
225 if ((cmsprofiles
= HeapAlloc( GetProcessHeap(), 0, (nprofiles
+ 1) * sizeof(cmsHPROFILE
) )))
227 cmsprofiles
[0] = profile0
->cmsprofile
;
228 cmsprofiles
[1] = profile1
->cmsprofile
;
230 transform
.cmstransform
= cmsCreateMultiprofileTransform( cmsprofiles
, nprofiles
, 0,
232 HeapFree( GetProcessHeap(), 0, cmsprofiles
);
233 if (!transform
.cmstransform
)
235 release_profile( profile0
);
236 release_profile( profile1
);
239 ret
= create_transform( &transform
);
242 release_profile( profile0
);
243 release_profile( profile1
);
245 #endif /* HAVE_LCMS2 */
249 /******************************************************************************
250 * DeleteColorTransform [MSCMS.@]
252 * Delete a color transform.
255 * transform [I] Handle to a color transform.
261 BOOL WINAPI
DeleteColorTransform( HTRANSFORM handle
)
266 TRACE( "( %p )\n", handle
);
268 ret
= close_transform( handle
);
270 #endif /* HAVE_LCMS2 */
274 /******************************************************************************
275 * TranslateBitmapBits [MSCMS.@]
277 * Perform color translation.
280 * transform [I] Handle to a color transform.
281 * srcbits [I] Source bitmap.
282 * input [I] Format of the source bitmap.
283 * width [I] Width of the source bitmap.
284 * height [I] Height of the source bitmap.
285 * inputstride [I] Number of bytes in one scanline.
286 * destbits [I] Destination bitmap.
287 * output [I] Format of the destination bitmap.
288 * outputstride [I] Number of bytes in one scanline.
289 * callback [I] Callback function.
290 * data [I] Callback data.
296 BOOL WINAPI
TranslateBitmapBits( HTRANSFORM handle
, PVOID srcbits
, BMFORMAT input
,
297 DWORD width
, DWORD height
, DWORD inputstride
, PVOID destbits
, BMFORMAT output
,
298 DWORD outputstride
, PBMCALLBACKFN callback
, ULONG data
)
302 struct transform
*transform
= grab_transform( handle
);
304 TRACE( "( %p, %p, 0x%08x, 0x%08x, 0x%08x, 0x%08x, %p, 0x%08x, 0x%08x, %p, 0x%08x )\n",
305 handle
, srcbits
, input
, width
, height
, inputstride
, destbits
, output
,
306 outputstride
, callback
, data
);
308 if (!transform
) return FALSE
;
309 if (!cmsChangeBuffersFormat( transform
->cmstransform
, from_bmformat(input
), from_bmformat(output
) ))
312 cmsDoTransform( transform
->cmstransform
, srcbits
, destbits
, width
* height
);
313 release_transform( transform
);
316 #endif /* HAVE_LCMS2 */
320 /******************************************************************************
321 * TranslateColors [MSCMS.@]
323 * Perform color translation.
326 * transform [I] Handle to a color transform.
327 * input [I] Array of input colors.
328 * number [I] Number of colors to translate.
329 * input_type [I] Input color format.
330 * output [O] Array of output colors.
331 * output_type [I] Output color format.
337 BOOL WINAPI
TranslateColors( HTRANSFORM handle
, PCOLOR in
, DWORD count
,
338 COLORTYPE input_type
, PCOLOR out
, COLORTYPE output_type
)
342 struct transform
*transform
= grab_transform( handle
);
346 TRACE( "( %p, %p, %d, %d, %p, %d )\n", handle
, in
, count
, input_type
, out
, output_type
);
348 if (!transform
) return FALSE
;
350 xfrm
= transform
->cmstransform
;
351 if (!cmsChangeBuffersFormat( xfrm
, from_type(input_type
), from_type(output_type
) ))
360 case COLOR_RGB
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].rgb
, &out
[i
].rgb
, 1 ); goto done
;
361 case COLOR_Lab
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].rgb
, &out
[i
].Lab
, 1 ); goto done
;
362 case COLOR_GRAY
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].rgb
, &out
[i
].gray
, 1 ); goto done
;
363 case COLOR_CMYK
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].rgb
, &out
[i
].cmyk
, 1 ); goto done
;
364 case COLOR_XYZ
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].rgb
, &out
[i
].XYZ
, 1 ); goto done
;
366 FIXME("unhandled input/output pair: %d/%d\n", input_type
, output_type
);
376 case COLOR_RGB
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].Lab
, &out
[i
].rgb
, 1 ); goto done
;
377 case COLOR_Lab
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].Lab
, &out
[i
].Lab
, 1 ); goto done
;
378 case COLOR_GRAY
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].Lab
, &out
[i
].gray
, 1 ); goto done
;
379 case COLOR_CMYK
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].Lab
, &out
[i
].cmyk
, 1 ); goto done
;
380 case COLOR_XYZ
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].Lab
, &out
[i
].XYZ
, 1 ); goto done
;
382 FIXME("unhandled input/output pair: %d/%d\n", input_type
, output_type
);
392 case COLOR_RGB
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].gray
, &out
[i
].rgb
, 1 ); goto done
;
393 case COLOR_Lab
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].gray
, &out
[i
].Lab
, 1 ); goto done
;
394 case COLOR_GRAY
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].gray
, &out
[i
].gray
, 1 ); goto done
;
395 case COLOR_CMYK
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].gray
, &out
[i
].cmyk
, 1 ); goto done
;
396 case COLOR_XYZ
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].gray
, &out
[i
].XYZ
, 1 ); goto done
;
398 FIXME("unhandled input/output pair: %d/%d\n", input_type
, output_type
);
408 case COLOR_RGB
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].cmyk
, &out
[i
].rgb
, 1 ); goto done
;
409 case COLOR_Lab
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].cmyk
, &out
[i
].Lab
, 1 ); goto done
;
410 case COLOR_GRAY
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].cmyk
, &out
[i
].gray
, 1 ); goto done
;
411 case COLOR_CMYK
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].cmyk
, &out
[i
].cmyk
, 1 ); goto done
;
412 case COLOR_XYZ
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].cmyk
, &out
[i
].XYZ
, 1 ); goto done
;
414 FIXME("unhandled input/output pair: %d/%d\n", input_type
, output_type
);
424 case COLOR_RGB
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].XYZ
, &out
[i
].rgb
, 1 ); goto done
;
425 case COLOR_Lab
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].XYZ
, &out
[i
].Lab
, 1 ); goto done
;
426 case COLOR_GRAY
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].XYZ
, &out
[i
].gray
, 1 ); goto done
;
427 case COLOR_CMYK
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].XYZ
, &out
[i
].cmyk
, 1 ); goto done
;
428 case COLOR_XYZ
: for (i
= 0; i
< count
; i
++) cmsDoTransform( xfrm
, &in
[i
].XYZ
, &out
[i
].XYZ
, 1 ); goto done
;
430 FIXME("unhandled input/output pair: %d/%d\n", input_type
, output_type
);
437 FIXME("unhandled input/output pair: %d/%d\n", input_type
, output_type
);
443 release_transform( transform
);
446 #else /* HAVE_LCMS2 */
448 #endif /* HAVE_LCMS2 */