1 //---------------------------------------------------------------------------------
2 //
3 //  Little Color Management System
4 //  Copyright (c) 1998-2016 Marti Maria Saguer
5 //
6 // Permission is hereby granted, free of charge, to any person obtaining
7 // a copy of this software and associated documentation files (the "Software"),
8 // to deal in the Software without restriction, including without limitation
9 // the rights to use, copy, modify, merge, publish, distribute, sublicense,
10 // and/or sell copies of the Software, and to permit persons to whom the Software
11 // is furnished to do so, subject to the following conditions:
12 //
13 // The above copyright notice and this permission notice shall be included in
14 // all copies or substantial portions of the Software.
15 //
16 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO
18 // THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23 //
24 //---------------------------------------------------------------------------------
25 //
26 
27 #include "lcms2_internal.h"
28 
29 // Multilocalized unicode objects. That is an attempt to encapsulate i18n.
30 
31 
32 // Allocates an empty multi localizad unicode object
cmsMLUalloc(cmsContext ContextID,cmsUInt32Number nItems)33 cmsMLU* CMSEXPORT cmsMLUalloc(cmsContext ContextID, cmsUInt32Number nItems)
34 {
35     cmsMLU* mlu;
36 
37     // nItems should be positive if given
38     if (nItems <= 0) nItems = 2;
39 
40     // Create the container
41     mlu = (cmsMLU*) _cmsMallocZero(ContextID, sizeof(cmsMLU));
42     if (mlu == NULL) return NULL;
43 
44     mlu ->ContextID = ContextID;
45 
46     // Create entry array
47     mlu ->Entries = (_cmsMLUentry*) _cmsCalloc(ContextID, nItems, sizeof(_cmsMLUentry));
48     if (mlu ->Entries == NULL) {
49         _cmsFree(ContextID, mlu);
50         return NULL;
51     }
52 
53     // Ok, keep indexes up to date
54     mlu ->AllocatedEntries    = nItems;
55     mlu ->UsedEntries         = 0;
56 
57     return mlu;
58 }
59 
60 
61 // Grows a mempool table for a MLU. Each time this function is called, mempool size is multiplied times two.
62 static
GrowMLUpool(cmsMLU * mlu)63 cmsBool GrowMLUpool(cmsMLU* mlu)
64 {
65     cmsUInt32Number size;
66     void *NewPtr;
67 
68     // Sanity check
69     if (mlu == NULL) return FALSE;
70 
71     if (mlu ->PoolSize == 0)
72         size = 256;
73     else
74         size = mlu ->PoolSize * 2;
75 
76     // Check for overflow
77     if (size < mlu ->PoolSize) return FALSE;
78 
79     // Reallocate the pool
80     NewPtr = _cmsRealloc(mlu ->ContextID, mlu ->MemPool, size);
81     if (NewPtr == NULL) return FALSE;
82 
83 
84     mlu ->MemPool  = NewPtr;
85     mlu ->PoolSize = size;
86 
87     return TRUE;
88 }
89 
90 
91 // Grows a entry table for a MLU. Each time this function is called, table size is multiplied times two.
92 static
GrowMLUtable(cmsMLU * mlu)93 cmsBool GrowMLUtable(cmsMLU* mlu)
94 {
95     cmsUInt32Number AllocatedEntries;
96     _cmsMLUentry *NewPtr;
97 
98     // Sanity check
99     if (mlu == NULL) return FALSE;
100 
101     AllocatedEntries = mlu ->AllocatedEntries * 2;
102 
103     // Check for overflow
104     if (AllocatedEntries / 2 != mlu ->AllocatedEntries) return FALSE;
105 
106     // Reallocate the memory
107     NewPtr = (_cmsMLUentry*)_cmsRealloc(mlu ->ContextID, mlu ->Entries, AllocatedEntries*sizeof(_cmsMLUentry));
108     if (NewPtr == NULL) return FALSE;
109 
110     mlu ->Entries          = NewPtr;
111     mlu ->AllocatedEntries = AllocatedEntries;
112 
113     return TRUE;
114 }
115 
116 
117 // Search for a specific entry in the structure. Language and Country are used.
118 static
SearchMLUEntry(cmsMLU * mlu,cmsUInt16Number LanguageCode,cmsUInt16Number CountryCode)119 int SearchMLUEntry(cmsMLU* mlu, cmsUInt16Number LanguageCode, cmsUInt16Number CountryCode)
120 {
121     cmsUInt32Number i;
122 
123     // Sanity check
124     if (mlu == NULL) return -1;
125 
126     // Iterate whole table
127     for (i=0; i < mlu ->UsedEntries; i++) {
128 
129         if (mlu ->Entries[i].Country  == CountryCode &&
130             mlu ->Entries[i].Language == LanguageCode) return i;
131     }
132 
133     // Not found
134     return -1;
135 }
136 
137 // Add a block of characters to the intended MLU. Language and country are specified.
138 // Only one entry for Language/country pair is allowed.
139 static
AddMLUBlock(cmsMLU * mlu,cmsUInt32Number size,const wchar_t * Block,cmsUInt16Number LanguageCode,cmsUInt16Number CountryCode)140 cmsBool AddMLUBlock(cmsMLU* mlu, cmsUInt32Number size, const wchar_t *Block,
141                      cmsUInt16Number LanguageCode, cmsUInt16Number CountryCode)
142 {
143     cmsUInt32Number Offset;
144     cmsUInt8Number* Ptr;
145 
146     // Sanity check
147     if (mlu == NULL) return FALSE;
148 
149     // Is there any room available?
150     if (mlu ->UsedEntries >= mlu ->AllocatedEntries) {
151         if (!GrowMLUtable(mlu)) return FALSE;
152     }
153 
154     // Only one ASCII string
155     if (SearchMLUEntry(mlu, LanguageCode, CountryCode) >= 0) return FALSE;  // Only one  is allowed!
156 
157     // Check for size
158     while ((mlu ->PoolSize - mlu ->PoolUsed) < size) {
159 
160             if (!GrowMLUpool(mlu)) return FALSE;
161     }
162 
163     Offset = mlu ->PoolUsed;
164 
165     Ptr = (cmsUInt8Number*) mlu ->MemPool;
166     if (Ptr == NULL) return FALSE;
167 
168     // Set the entry
169     memmove(Ptr + Offset, Block, size);
170     mlu ->PoolUsed += size;
171 
172     mlu ->Entries[mlu ->UsedEntries].StrW     = Offset;
173     mlu ->Entries[mlu ->UsedEntries].Len      = size;
174     mlu ->Entries[mlu ->UsedEntries].Country  = CountryCode;
175     mlu ->Entries[mlu ->UsedEntries].Language = LanguageCode;
176     mlu ->UsedEntries++;
177 
178     return TRUE;
179 }
180 
181 // Convert from a 3-char code to a cmsUInt16Number. It is done inthis way because some
182 // compilers don't properly align beginning of strings
183 
184 static
strTo16(const char str[3])185 cmsUInt16Number strTo16(const char str[3])
186 {
187     cmsUInt16Number n = ((cmsUInt16Number) str[0] << 8) | str[1];
188 
189     return n;  // Always big endian in this case
190 }
191 
192 static
strFrom16(char str[3],cmsUInt16Number n)193 void strFrom16(char str[3], cmsUInt16Number n)
194 {
195     // Assiming this would be aligned
196     union {
197 
198        cmsUInt16Number n;
199        char str[2];
200 
201     } c;
202 
203     c.n = n;  // Always big endian in this case
204 
205     str[0] = c.str[0]; str[1] = c.str[1]; str[2] = 0;
206 
207 }
208 
209 // Add an ASCII entry. Do not add any \0 termination (ICC1v43_2010-12.pdf page 61)
cmsMLUsetASCII(cmsMLU * mlu,const char LanguageCode[3],const char CountryCode[3],const char * ASCIIString)210 cmsBool CMSEXPORT cmsMLUsetASCII(cmsMLU* mlu, const char LanguageCode[3], const char CountryCode[3], const char* ASCIIString)
211 {
212     cmsUInt32Number i, len = (cmsUInt32Number) strlen(ASCIIString);
213     wchar_t* WStr;
214     cmsBool  rc;
215     cmsUInt16Number Lang  = strTo16(LanguageCode);
216     cmsUInt16Number Cntry = strTo16(CountryCode);
217 
218     if (mlu == NULL) return FALSE;
219 
220     WStr = (wchar_t*) _cmsCalloc(mlu ->ContextID, len,  sizeof(wchar_t));
221     if (WStr == NULL) return FALSE;
222 
223     for (i=0; i < len; i++)
224         WStr[i] = (wchar_t) ASCIIString[i];
225 
226     rc = AddMLUBlock(mlu, len  * sizeof(wchar_t), WStr, Lang, Cntry);
227 
228     _cmsFree(mlu ->ContextID, WStr);
229     return rc;
230 
231 }
232 
233 // We don't need any wcs support library
234 static
mywcslen(const wchar_t * s)235 cmsUInt32Number mywcslen(const wchar_t *s)
236 {
237     const wchar_t *p;
238 
239     p = s;
240     while (*p)
241         p++;
242 
243     return (cmsUInt32Number)(p - s);
244 }
245 
246 // Add a wide entry. Do not add any \0 terminator (ICC1v43_2010-12.pdf page 61)
cmsMLUsetWide(cmsMLU * mlu,const char Language[3],const char Country[3],const wchar_t * WideString)247 cmsBool  CMSEXPORT cmsMLUsetWide(cmsMLU* mlu, const char Language[3], const char Country[3], const wchar_t* WideString)
248 {
249     cmsUInt16Number Lang  = strTo16(Language);
250     cmsUInt16Number Cntry = strTo16(Country);
251     cmsUInt32Number len;
252 
253     if (mlu == NULL) return FALSE;
254     if (WideString == NULL) return FALSE;
255 
256     len = (cmsUInt32Number) (mywcslen(WideString)) * sizeof(wchar_t);
257     return AddMLUBlock(mlu, len, WideString, Lang, Cntry);
258 }
259 
260 // Duplicating a MLU is as easy as copying all members
cmsMLUdup(const cmsMLU * mlu)261 cmsMLU* CMSEXPORT cmsMLUdup(const cmsMLU* mlu)
262 {
263     cmsMLU* NewMlu = NULL;
264 
265     // Duplicating a NULL obtains a NULL
266     if (mlu == NULL) return NULL;
267 
268     NewMlu = cmsMLUalloc(mlu ->ContextID, mlu ->UsedEntries);
269     if (NewMlu == NULL) return NULL;
270 
271     // Should never happen
272     if (NewMlu ->AllocatedEntries < mlu ->UsedEntries)
273         goto Error;
274 
275     // Sanitize...
276     if (NewMlu ->Entries == NULL || mlu ->Entries == NULL)  goto Error;
277 
278     memmove(NewMlu ->Entries, mlu ->Entries, mlu ->UsedEntries * sizeof(_cmsMLUentry));
279     NewMlu ->UsedEntries = mlu ->UsedEntries;
280 
281     // The MLU may be empty
282     if (mlu ->PoolUsed == 0) {
283         NewMlu ->MemPool = NULL;
284     }
285     else {
286         // It is not empty
287         NewMlu ->MemPool = _cmsMalloc(mlu ->ContextID, mlu ->PoolUsed);
288         if (NewMlu ->MemPool == NULL) goto Error;
289     }
290 
291     NewMlu ->PoolSize = mlu ->PoolUsed;
292 
293     if (NewMlu ->MemPool == NULL || mlu ->MemPool == NULL) goto Error;
294 
295     memmove(NewMlu ->MemPool, mlu->MemPool, mlu ->PoolUsed);
296     NewMlu ->PoolUsed = mlu ->PoolUsed;
297 
298     return NewMlu;
299 
300 Error:
301 
302     if (NewMlu != NULL) cmsMLUfree(NewMlu);
303     return NULL;
304 }
305 
306 // Free any used memory
cmsMLUfree(cmsMLU * mlu)307 void CMSEXPORT cmsMLUfree(cmsMLU* mlu)
308 {
309     if (mlu) {
310 
311         if (mlu -> Entries) _cmsFree(mlu ->ContextID, mlu->Entries);
312         if (mlu -> MemPool) _cmsFree(mlu ->ContextID, mlu->MemPool);
313 
314         _cmsFree(mlu ->ContextID, mlu);
315     }
316 }
317 
318 
319 // The algorithm first searches for an exact match of country and language, if not found it uses
320 // the Language. If none is found, first entry is used instead.
321 static
_cmsMLUgetWide(const cmsMLU * mlu,cmsUInt32Number * len,cmsUInt16Number LanguageCode,cmsUInt16Number CountryCode,cmsUInt16Number * UsedLanguageCode,cmsUInt16Number * UsedCountryCode)322 const wchar_t* _cmsMLUgetWide(const cmsMLU* mlu,
323                               cmsUInt32Number *len,
324                               cmsUInt16Number LanguageCode, cmsUInt16Number CountryCode,
325                               cmsUInt16Number* UsedLanguageCode, cmsUInt16Number* UsedCountryCode)
326 {
327     cmsUInt32Number i;
328     cmsInt32Number Best = -1;
329     _cmsMLUentry* v;
330 
331     if (mlu == NULL) return NULL;
332 
333     if (mlu -> AllocatedEntries <= 0) return NULL;
334 
335     for (i=0; i < mlu ->UsedEntries; i++) {
336 
337         v = mlu ->Entries + i;
338 
339         if (v -> Language == LanguageCode) {
340 
341             if (Best == -1) Best = i;
342 
343             if (v -> Country == CountryCode) {
344 
345                 if (UsedLanguageCode != NULL) *UsedLanguageCode = v ->Language;
346                 if (UsedCountryCode  != NULL) *UsedCountryCode = v ->Country;
347 
348                 if (len != NULL) *len = v ->Len;
349 
350                 return (wchar_t*) ((cmsUInt8Number*) mlu ->MemPool + v -> StrW);        // Found exact match
351             }
352         }
353     }
354 
355     // No string found. Return First one
356     if (Best == -1)
357         Best = 0;
358 
359     v = mlu ->Entries + Best;
360 
361     if (UsedLanguageCode != NULL) *UsedLanguageCode = v ->Language;
362     if (UsedCountryCode  != NULL) *UsedCountryCode = v ->Country;
363 
364     if (len != NULL) *len   = v ->Len;
365 
366     return(wchar_t*) ((cmsUInt8Number*) mlu ->MemPool + v ->StrW);
367 }
368 
369 
370 // Obtain an ASCII representation of the wide string. Setting buffer to NULL returns the len
cmsMLUgetASCII(const cmsMLU * mlu,const char LanguageCode[3],const char CountryCode[3],char * Buffer,cmsUInt32Number BufferSize)371 cmsUInt32Number CMSEXPORT cmsMLUgetASCII(const cmsMLU* mlu,
372                                        const char LanguageCode[3], const char CountryCode[3],
373                                        char* Buffer, cmsUInt32Number BufferSize)
374 {
375     const wchar_t *Wide;
376     cmsUInt32Number  StrLen = 0;
377     cmsUInt32Number ASCIIlen, i;
378 
379     cmsUInt16Number Lang  = strTo16(LanguageCode);
380     cmsUInt16Number Cntry = strTo16(CountryCode);
381 
382     // Sanitize
383     if (mlu == NULL) return 0;
384 
385     // Get WideChar
386     Wide = _cmsMLUgetWide(mlu, &StrLen, Lang, Cntry, NULL, NULL);
387     if (Wide == NULL) return 0;
388 
389     ASCIIlen = StrLen / sizeof(wchar_t);
390 
391     // Maybe we want only to know the len?
392     if (Buffer == NULL) return ASCIIlen + 1; // Note the zero at the end
393 
394     // No buffer size means no data
395     if (BufferSize <= 0) return 0;
396 
397     // Some clipping may be required
398     if (BufferSize < ASCIIlen + 1)
399         ASCIIlen = BufferSize - 1;
400 
401     // Precess each character
402     for (i=0; i < ASCIIlen; i++) {
403 
404         if (Wide[i] == 0)
405             Buffer[i] = 0;
406         else
407             Buffer[i] = (char) Wide[i];
408     }
409 
410     // We put a termination "\0"
411     Buffer[ASCIIlen] = 0;
412     return ASCIIlen + 1;
413 }
414 
415 // Obtain a wide representation of the MLU, on depending on current locale settings
cmsMLUgetWide(const cmsMLU * mlu,const char LanguageCode[3],const char CountryCode[3],wchar_t * Buffer,cmsUInt32Number BufferSize)416 cmsUInt32Number CMSEXPORT cmsMLUgetWide(const cmsMLU* mlu,
417                                       const char LanguageCode[3], const char CountryCode[3],
418                                       wchar_t* Buffer, cmsUInt32Number BufferSize)
419 {
420     const wchar_t *Wide;
421     cmsUInt32Number  StrLen = 0;
422 
423     cmsUInt16Number Lang  = strTo16(LanguageCode);
424     cmsUInt16Number Cntry = strTo16(CountryCode);
425 
426     // Sanitize
427     if (mlu == NULL) return 0;
428 
429     Wide = _cmsMLUgetWide(mlu, &StrLen, Lang, Cntry, NULL, NULL);
430     if (Wide == NULL) return 0;
431 
432     // Maybe we want only to know the len?
433     if (Buffer == NULL) return StrLen + sizeof(wchar_t);
434 
435   // No buffer size means no data
436     if (BufferSize <= 0) return 0;
437 
438     // Some clipping may be required
439     if (BufferSize < StrLen + sizeof(wchar_t))
440         StrLen = BufferSize - + sizeof(wchar_t);
441 
442     memmove(Buffer, Wide, StrLen);
443     Buffer[StrLen / sizeof(wchar_t)] = 0;
444 
445     return StrLen + sizeof(wchar_t);
446 }
447 
448 
449 // Get also the language and country
cmsMLUgetTranslation(const cmsMLU * mlu,const char LanguageCode[3],const char CountryCode[3],char ObtainedLanguage[3],char ObtainedCountry[3])450 CMSAPI cmsBool CMSEXPORT cmsMLUgetTranslation(const cmsMLU* mlu,
451                                               const char LanguageCode[3], const char CountryCode[3],
452                                               char ObtainedLanguage[3], char ObtainedCountry[3])
453 {
454     const wchar_t *Wide;
455 
456     cmsUInt16Number Lang  = strTo16(LanguageCode);
457     cmsUInt16Number Cntry = strTo16(CountryCode);
458     cmsUInt16Number ObtLang, ObtCode;
459 
460     // Sanitize
461     if (mlu == NULL) return FALSE;
462 
463     Wide = _cmsMLUgetWide(mlu, NULL, Lang, Cntry, &ObtLang, &ObtCode);
464     if (Wide == NULL) return FALSE;
465 
466     // Get used language and code
467     strFrom16(ObtainedLanguage, ObtLang);
468     strFrom16(ObtainedCountry, ObtCode);
469 
470     return TRUE;
471 }
472 
473 
474 
475 // Get the number of translations in the MLU object
cmsMLUtranslationsCount(const cmsMLU * mlu)476 cmsUInt32Number CMSEXPORT cmsMLUtranslationsCount(const cmsMLU* mlu)
477 {
478     if (mlu == NULL) return 0;
479     return mlu->UsedEntries;
480 }
481 
482 // Get the language and country codes for a specific MLU index
cmsMLUtranslationsCodes(const cmsMLU * mlu,cmsUInt32Number idx,char LanguageCode[3],char CountryCode[3])483 cmsBool CMSEXPORT cmsMLUtranslationsCodes(const cmsMLU* mlu,
484                                           cmsUInt32Number idx,
485                                           char LanguageCode[3],
486                                           char CountryCode[3])
487 {
488     _cmsMLUentry *entry;
489 
490     if (mlu == NULL) return FALSE;
491 
492     if (idx >= mlu->UsedEntries) return FALSE;
493 
494     entry = &mlu->Entries[idx];
495 
496     strFrom16(LanguageCode, entry->Language);
497     strFrom16(CountryCode, entry->Country);
498 
499     return TRUE;
500 }
501 
502 
503 // Named color lists --------------------------------------------------------------------------------------------
504 
505 // Grow the list to keep at least NumElements
506 static
GrowNamedColorList(cmsNAMEDCOLORLIST * v)507 cmsBool  GrowNamedColorList(cmsNAMEDCOLORLIST* v)
508 {
509     cmsUInt32Number size;
510     _cmsNAMEDCOLOR * NewPtr;
511 
512     if (v == NULL) return FALSE;
513 
514     if (v ->Allocated == 0)
515         size = 64;   // Initial guess
516     else
517         size = v ->Allocated * 2;
518 
519     // Keep a maximum color lists can grow, 100K entries seems reasonable
520     if (size > 1024*100) return FALSE;
521 
522     NewPtr = (_cmsNAMEDCOLOR*) _cmsRealloc(v ->ContextID, v ->List, size * sizeof(_cmsNAMEDCOLOR));
523     if (NewPtr == NULL)
524         return FALSE;
525 
526     v ->List      = NewPtr;
527     v ->Allocated = size;
528     return TRUE;
529 }
530 
531 // Allocate a list for n elements
cmsAllocNamedColorList(cmsContext ContextID,cmsUInt32Number n,cmsUInt32Number ColorantCount,const char * Prefix,const char * Suffix)532 cmsNAMEDCOLORLIST* CMSEXPORT cmsAllocNamedColorList(cmsContext ContextID, cmsUInt32Number n, cmsUInt32Number ColorantCount, const char* Prefix, const char* Suffix)
533 {
534     cmsNAMEDCOLORLIST* v = (cmsNAMEDCOLORLIST*) _cmsMallocZero(ContextID, sizeof(cmsNAMEDCOLORLIST));
535 
536     if (v == NULL) return NULL;
537 
538     v ->List      = NULL;
539     v ->nColors   = 0;
540     v ->ContextID  = ContextID;
541 
542     while (v -> Allocated < n){
543         if (!GrowNamedColorList(v)) {
544             cmsFreeNamedColorList(v);
545             return NULL;
546         }
547     }
548 
549     strncpy(v ->Prefix, Prefix, sizeof(v ->Prefix)-1);
550     strncpy(v ->Suffix, Suffix, sizeof(v ->Suffix)-1);
551     v->Prefix[32] = v->Suffix[32] = 0;
552 
553     v -> ColorantCount = ColorantCount;
554 
555     return v;
556 }
557 
558 // Free a list
cmsFreeNamedColorList(cmsNAMEDCOLORLIST * v)559 void CMSEXPORT cmsFreeNamedColorList(cmsNAMEDCOLORLIST* v)
560 {
561     if (v == NULL) return;
562     if (v ->List) _cmsFree(v ->ContextID, v ->List);
563     _cmsFree(v ->ContextID, v);
564 }
565 
cmsDupNamedColorList(const cmsNAMEDCOLORLIST * v)566 cmsNAMEDCOLORLIST* CMSEXPORT cmsDupNamedColorList(const cmsNAMEDCOLORLIST* v)
567 {
568     cmsNAMEDCOLORLIST* NewNC;
569 
570     if (v == NULL) return NULL;
571 
572     NewNC= cmsAllocNamedColorList(v ->ContextID, v -> nColors, v ->ColorantCount, v ->Prefix, v ->Suffix);
573     if (NewNC == NULL) return NULL;
574 
575     // For really large tables we need this
576     while (NewNC ->Allocated < v ->Allocated){
577         if (!GrowNamedColorList(NewNC)) {
578             cmsFreeNamedColorList(NewNC);
579             return NULL;
580         }
581     }
582 
583     memmove(NewNC ->Prefix, v ->Prefix, sizeof(v ->Prefix));
584     memmove(NewNC ->Suffix, v ->Suffix, sizeof(v ->Suffix));
585     NewNC ->ColorantCount = v ->ColorantCount;
586     memmove(NewNC->List, v ->List, v->nColors * sizeof(_cmsNAMEDCOLOR));
587     NewNC ->nColors = v ->nColors;
588     return NewNC;
589 }
590 
591 
592 // Append a color to a list. List pointer may change if reallocated
cmsAppendNamedColor(cmsNAMEDCOLORLIST * NamedColorList,const char * Name,cmsUInt16Number PCS[3],cmsUInt16Number Colorant[cmsMAXCHANNELS])593 cmsBool  CMSEXPORT cmsAppendNamedColor(cmsNAMEDCOLORLIST* NamedColorList,
594                                        const char* Name,
595                                        cmsUInt16Number PCS[3], cmsUInt16Number Colorant[cmsMAXCHANNELS])
596 {
597     cmsUInt32Number i;
598 
599     if (NamedColorList == NULL) return FALSE;
600 
601     if (NamedColorList ->nColors + 1 > NamedColorList ->Allocated) {
602         if (!GrowNamedColorList(NamedColorList)) return FALSE;
603     }
604 
605     for (i=0; i < NamedColorList ->ColorantCount; i++)
606         NamedColorList ->List[NamedColorList ->nColors].DeviceColorant[i] = Colorant == NULL? 0 : Colorant[i];
607 
608     for (i=0; i < 3; i++)
609         NamedColorList ->List[NamedColorList ->nColors].PCS[i] = PCS == NULL ? 0 : PCS[i];
610 
611     if (Name != NULL) {
612 
613         strncpy(NamedColorList ->List[NamedColorList ->nColors].Name, Name, cmsMAX_PATH-1);
614         NamedColorList ->List[NamedColorList ->nColors].Name[cmsMAX_PATH-1] = 0;
615 
616     }
617     else
618         NamedColorList ->List[NamedColorList ->nColors].Name[0] = 0;
619 
620 
621     NamedColorList ->nColors++;
622     return TRUE;
623 }
624 
625 // Returns number of elements
cmsNamedColorCount(const cmsNAMEDCOLORLIST * NamedColorList)626 cmsUInt32Number CMSEXPORT cmsNamedColorCount(const cmsNAMEDCOLORLIST* NamedColorList)
627 {
628      if (NamedColorList == NULL) return 0;
629      return NamedColorList ->nColors;
630 }
631 
632 // Info aboout a given color
cmsNamedColorInfo(const cmsNAMEDCOLORLIST * NamedColorList,cmsUInt32Number nColor,char * Name,char * Prefix,char * Suffix,cmsUInt16Number * PCS,cmsUInt16Number * Colorant)633 cmsBool  CMSEXPORT cmsNamedColorInfo(const cmsNAMEDCOLORLIST* NamedColorList, cmsUInt32Number nColor,
634                                      char* Name,
635                                      char* Prefix,
636                                      char* Suffix,
637                                      cmsUInt16Number* PCS,
638                                      cmsUInt16Number* Colorant)
639 {
640     if (NamedColorList == NULL) return FALSE;
641 
642     if (nColor >= cmsNamedColorCount(NamedColorList)) return FALSE;
643 
644     if (Name) strcpy(Name, NamedColorList->List[nColor].Name);
645     if (Prefix) strcpy(Prefix, NamedColorList->Prefix);
646     if (Suffix) strcpy(Suffix, NamedColorList->Suffix);
647     if (PCS)
648         memmove(PCS, NamedColorList ->List[nColor].PCS, 3*sizeof(cmsUInt16Number));
649 
650     if (Colorant)
651         memmove(Colorant, NamedColorList ->List[nColor].DeviceColorant,
652                                 sizeof(cmsUInt16Number) * NamedColorList ->ColorantCount);
653 
654 
655     return TRUE;
656 }
657 
658 // Search for a given color name (no prefix or suffix)
cmsNamedColorIndex(const cmsNAMEDCOLORLIST * NamedColorList,const char * Name)659 cmsInt32Number CMSEXPORT cmsNamedColorIndex(const cmsNAMEDCOLORLIST* NamedColorList, const char* Name)
660 {
661     int i, n;
662 
663     if (NamedColorList == NULL) return -1;
664     n = cmsNamedColorCount(NamedColorList);
665     for (i=0; i < n; i++) {
666         if (cmsstrcasecmp(Name,  NamedColorList->List[i].Name) == 0)
667             return i;
668     }
669 
670     return -1;
671 }
672 
673 // MPE support -----------------------------------------------------------------------------------------------------------------
674 
675 static
FreeNamedColorList(cmsStage * mpe)676 void FreeNamedColorList(cmsStage* mpe)
677 {
678     cmsNAMEDCOLORLIST* List = (cmsNAMEDCOLORLIST*) mpe ->Data;
679     cmsFreeNamedColorList(List);
680 }
681 
682 static
DupNamedColorList(cmsStage * mpe)683 void* DupNamedColorList(cmsStage* mpe)
684 {
685     cmsNAMEDCOLORLIST* List = (cmsNAMEDCOLORLIST*) mpe ->Data;
686     return cmsDupNamedColorList(List);
687 }
688 
689 static
EvalNamedColorPCS(const cmsFloat32Number In[],cmsFloat32Number Out[],const cmsStage * mpe)690 void EvalNamedColorPCS(const cmsFloat32Number In[], cmsFloat32Number Out[], const cmsStage *mpe)
691 {
692     cmsNAMEDCOLORLIST* NamedColorList = (cmsNAMEDCOLORLIST*) mpe ->Data;
693     cmsUInt16Number index = (cmsUInt16Number) _cmsQuickSaturateWord(In[0] * 65535.0);
694 
695     if (index >= NamedColorList-> nColors) {
696         cmsSignalError(NamedColorList ->ContextID, cmsERROR_RANGE, "Color %d out of range; ignored", index);
697     }
698     else {
699 
700             // Named color always uses Lab
701             Out[0] = (cmsFloat32Number) (NamedColorList->List[index].PCS[0] / 65535.0);
702             Out[1] = (cmsFloat32Number) (NamedColorList->List[index].PCS[1] / 65535.0);
703             Out[2] = (cmsFloat32Number) (NamedColorList->List[index].PCS[2] / 65535.0);
704     }
705 }
706 
707 static
EvalNamedColor(const cmsFloat32Number In[],cmsFloat32Number Out[],const cmsStage * mpe)708 void EvalNamedColor(const cmsFloat32Number In[], cmsFloat32Number Out[], const cmsStage *mpe)
709 {
710     cmsNAMEDCOLORLIST* NamedColorList = (cmsNAMEDCOLORLIST*) mpe ->Data;
711     cmsUInt16Number index = (cmsUInt16Number) _cmsQuickSaturateWord(In[0] * 65535.0);
712     cmsUInt32Number j;
713 
714     if (index >= NamedColorList-> nColors) {
715         cmsSignalError(NamedColorList ->ContextID, cmsERROR_RANGE, "Color %d out of range; ignored", index);
716     }
717     else {
718         for (j=0; j < NamedColorList ->ColorantCount; j++)
719             Out[j] = (cmsFloat32Number) (NamedColorList->List[index].DeviceColorant[j] / 65535.0);
720     }
721 }
722 
723 
724 // Named color lookup element
_cmsStageAllocNamedColor(cmsNAMEDCOLORLIST * NamedColorList,cmsBool UsePCS)725 cmsStage* _cmsStageAllocNamedColor(cmsNAMEDCOLORLIST* NamedColorList, cmsBool UsePCS)
726 {
727     return _cmsStageAllocPlaceholder(NamedColorList ->ContextID,
728                                    cmsSigNamedColorElemType,
729                                    1, UsePCS ? 3 : NamedColorList ->ColorantCount,
730                                    UsePCS ? EvalNamedColorPCS : EvalNamedColor,
731                                    DupNamedColorList,
732                                    FreeNamedColorList,
733                                    cmsDupNamedColorList(NamedColorList));
734 
735 }
736 
737 
738 // Retrieve the named color list from a transform. Should be first element in the LUT
cmsGetNamedColorList(cmsHTRANSFORM xform)739 cmsNAMEDCOLORLIST* CMSEXPORT cmsGetNamedColorList(cmsHTRANSFORM xform)
740 {
741     _cmsTRANSFORM* v = (_cmsTRANSFORM*) xform;
742     cmsStage* mpe  = v ->Lut->Elements;
743 
744     if (mpe ->Type != cmsSigNamedColorElemType) return NULL;
745     return (cmsNAMEDCOLORLIST*) mpe ->Data;
746 }
747 
748 
749 // Profile sequence description routines -------------------------------------------------------------------------------------
750 
cmsAllocProfileSequenceDescription(cmsContext ContextID,cmsUInt32Number n)751 cmsSEQ* CMSEXPORT cmsAllocProfileSequenceDescription(cmsContext ContextID, cmsUInt32Number n)
752 {
753     cmsSEQ* Seq;
754     cmsUInt32Number i;
755 
756     if (n == 0) return NULL;
757 
758     // In a absolutely arbitrary way, I hereby decide to allow a maxim of 255 profiles linked
759     // in a devicelink. It makes not sense anyway and may be used for exploits, so let's close the door!
760     if (n > 255) return NULL;
761 
762     Seq = (cmsSEQ*) _cmsMallocZero(ContextID, sizeof(cmsSEQ));
763     if (Seq == NULL) return NULL;
764 
765     Seq -> ContextID = ContextID;
766     Seq -> seq      = (cmsPSEQDESC*) _cmsCalloc(ContextID, n, sizeof(cmsPSEQDESC));
767     Seq -> n        = n;
768 
769     if (Seq -> seq == NULL) {
770         _cmsFree(ContextID, Seq);
771         return NULL;
772     }
773 
774     for (i=0; i < n; i++) {
775         Seq -> seq[i].Manufacturer = NULL;
776         Seq -> seq[i].Model        = NULL;
777         Seq -> seq[i].Description  = NULL;
778     }
779 
780     return Seq;
781 }
782 
cmsFreeProfileSequenceDescription(cmsSEQ * pseq)783 void CMSEXPORT cmsFreeProfileSequenceDescription(cmsSEQ* pseq)
784 {
785     cmsUInt32Number i;
786 
787     for (i=0; i < pseq ->n; i++) {
788         if (pseq ->seq[i].Manufacturer != NULL) cmsMLUfree(pseq ->seq[i].Manufacturer);
789         if (pseq ->seq[i].Model != NULL) cmsMLUfree(pseq ->seq[i].Model);
790         if (pseq ->seq[i].Description != NULL) cmsMLUfree(pseq ->seq[i].Description);
791     }
792 
793     if (pseq ->seq != NULL) _cmsFree(pseq ->ContextID, pseq ->seq);
794     _cmsFree(pseq -> ContextID, pseq);
795 }
796 
cmsDupProfileSequenceDescription(const cmsSEQ * pseq)797 cmsSEQ* CMSEXPORT cmsDupProfileSequenceDescription(const cmsSEQ* pseq)
798 {
799     cmsSEQ *NewSeq;
800     cmsUInt32Number i;
801 
802     if (pseq == NULL)
803         return NULL;
804 
805     NewSeq = (cmsSEQ*) _cmsMalloc(pseq -> ContextID, sizeof(cmsSEQ));
806     if (NewSeq == NULL) return NULL;
807 
808 
809     NewSeq -> seq      = (cmsPSEQDESC*) _cmsCalloc(pseq ->ContextID, pseq ->n, sizeof(cmsPSEQDESC));
810     if (NewSeq ->seq == NULL) goto Error;
811 
812     NewSeq -> ContextID = pseq ->ContextID;
813     NewSeq -> n        = pseq ->n;
814 
815     for (i=0; i < pseq->n; i++) {
816 
817         memmove(&NewSeq ->seq[i].attributes, &pseq ->seq[i].attributes, sizeof(cmsUInt64Number));
818 
819         NewSeq ->seq[i].deviceMfg   = pseq ->seq[i].deviceMfg;
820         NewSeq ->seq[i].deviceModel = pseq ->seq[i].deviceModel;
821         memmove(&NewSeq ->seq[i].ProfileID, &pseq ->seq[i].ProfileID, sizeof(cmsProfileID));
822         NewSeq ->seq[i].technology  = pseq ->seq[i].technology;
823 
824         NewSeq ->seq[i].Manufacturer = cmsMLUdup(pseq ->seq[i].Manufacturer);
825         NewSeq ->seq[i].Model        = cmsMLUdup(pseq ->seq[i].Model);
826         NewSeq ->seq[i].Description  = cmsMLUdup(pseq ->seq[i].Description);
827 
828     }
829 
830     return NewSeq;
831 
832 Error:
833 
834     cmsFreeProfileSequenceDescription(NewSeq);
835     return NULL;
836 }
837 
838 // Dictionaries --------------------------------------------------------------------------------------------------------
839 
840 // Dictionaries are just very simple linked lists
841 
842 
843 typedef struct _cmsDICT_struct {
844     cmsDICTentry* head;
845     cmsContext ContextID;
846 } _cmsDICT;
847 
848 
849 // Allocate an empty dictionary
cmsDictAlloc(cmsContext ContextID)850 cmsHANDLE CMSEXPORT cmsDictAlloc(cmsContext ContextID)
851 {
852     _cmsDICT* dict = (_cmsDICT*) _cmsMallocZero(ContextID, sizeof(_cmsDICT));
853     if (dict == NULL) return NULL;
854 
855     dict ->ContextID = ContextID;
856     return (cmsHANDLE) dict;
857 
858 }
859 
860 // Dispose resources
cmsDictFree(cmsHANDLE hDict)861 void CMSEXPORT cmsDictFree(cmsHANDLE hDict)
862 {
863     _cmsDICT* dict = (_cmsDICT*) hDict;
864     cmsDICTentry *entry, *next;
865 
866     _cmsAssert(dict != NULL);
867 
868     // Walk the list freeing all nodes
869     entry = dict ->head;
870     while (entry != NULL) {
871 
872             if (entry ->DisplayName  != NULL) cmsMLUfree(entry ->DisplayName);
873             if (entry ->DisplayValue != NULL) cmsMLUfree(entry ->DisplayValue);
874             if (entry ->Name != NULL) _cmsFree(dict ->ContextID, entry -> Name);
875             if (entry ->Value != NULL) _cmsFree(dict ->ContextID, entry -> Value);
876 
877             // Don't fall in the habitual trap...
878             next = entry ->Next;
879             _cmsFree(dict ->ContextID, entry);
880 
881             entry = next;
882     }
883 
884     _cmsFree(dict ->ContextID, dict);
885 }
886 
887 
888 // Duplicate a wide char string
889 static
DupWcs(cmsContext ContextID,const wchar_t * ptr)890 wchar_t* DupWcs(cmsContext ContextID, const wchar_t* ptr)
891 {
892     if (ptr == NULL) return NULL;
893     return (wchar_t*) _cmsDupMem(ContextID, ptr, (mywcslen(ptr) + 1) * sizeof(wchar_t));
894 }
895 
896 // Add a new entry to the linked list
cmsDictAddEntry(cmsHANDLE hDict,const wchar_t * Name,const wchar_t * Value,const cmsMLU * DisplayName,const cmsMLU * DisplayValue)897 cmsBool CMSEXPORT cmsDictAddEntry(cmsHANDLE hDict, const wchar_t* Name, const wchar_t* Value, const cmsMLU *DisplayName, const cmsMLU *DisplayValue)
898 {
899     _cmsDICT* dict = (_cmsDICT*) hDict;
900     cmsDICTentry *entry;
901 
902     _cmsAssert(dict != NULL);
903     _cmsAssert(Name != NULL);
904 
905     entry = (cmsDICTentry*) _cmsMallocZero(dict ->ContextID, sizeof(cmsDICTentry));
906     if (entry == NULL) return FALSE;
907 
908     entry ->DisplayName  = cmsMLUdup(DisplayName);
909     entry ->DisplayValue = cmsMLUdup(DisplayValue);
910     entry ->Name         = DupWcs(dict ->ContextID, Name);
911     entry ->Value        = DupWcs(dict ->ContextID, Value);
912 
913     entry ->Next = dict ->head;
914     dict ->head = entry;
915 
916     return TRUE;
917 }
918 
919 
920 // Duplicates an existing dictionary
cmsDictDup(cmsHANDLE hDict)921 cmsHANDLE CMSEXPORT cmsDictDup(cmsHANDLE hDict)
922 {
923     _cmsDICT* old_dict = (_cmsDICT*) hDict;
924     cmsHANDLE hNew;
925     cmsDICTentry *entry;
926 
927     _cmsAssert(old_dict != NULL);
928 
929     hNew  = cmsDictAlloc(old_dict ->ContextID);
930     if (hNew == NULL) return NULL;
931 
932     // Walk the list freeing all nodes
933     entry = old_dict ->head;
934     while (entry != NULL) {
935 
936         if (!cmsDictAddEntry(hNew, entry ->Name, entry ->Value, entry ->DisplayName, entry ->DisplayValue)) {
937 
938             cmsDictFree(hNew);
939             return NULL;
940         }
941 
942         entry = entry -> Next;
943     }
944 
945     return hNew;
946 }
947 
948 // Get a pointer to the linked list
cmsDictGetEntryList(cmsHANDLE hDict)949 const cmsDICTentry* CMSEXPORT cmsDictGetEntryList(cmsHANDLE hDict)
950 {
951     _cmsDICT* dict = (_cmsDICT*) hDict;
952 
953     if (dict == NULL) return NULL;
954     return dict ->head;
955 }
956 
957 // Helper For external languages
cmsDictNextEntry(const cmsDICTentry * e)958 const cmsDICTentry* CMSEXPORT cmsDictNextEntry(const cmsDICTentry* e)
959 {
960      if (e == NULL) return NULL;
961      return e ->Next;
962 }
963