OLD | NEW |
---|---|
1 /* | 1 /* |
2 * Copyright 2016 Google Inc. | 2 * Copyright 2016 Google Inc. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license that can be | 4 * Use of this source code is governed by a BSD-style license that can be |
5 * found in the LICENSE file. | 5 * found in the LICENSE file. |
6 */ | 6 */ |
7 | 7 |
8 #include "SkAtomics.h" | 8 #include "SkAtomics.h" |
9 #include "SkColorSpace.h" | 9 #include "SkColorSpace.h" |
10 #include "SkOncePtr.h" | 10 #include "SkOncePtr.h" |
(...skipping 304 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
315 } else if (len < 12 + 2 * count) { | 315 } else if (len < 12 + 2 * count) { |
316 SkColorSpacePrintf("gamma tag is too small (%d bytes)", len) ; | 316 SkColorSpacePrintf("gamma tag is too small (%d bytes)", len) ; |
317 return false; | 317 return false; |
318 } | 318 } |
319 | 319 |
320 const uint16_t* table = (const uint16_t*) (src + 12); | 320 const uint16_t* table = (const uint16_t*) (src + 12); |
321 if (1 == count) { | 321 if (1 == count) { |
322 // The table entry is the gamma (with a bias of 256). | 322 // The table entry is the gamma (with a bias of 256). |
323 uint16_t value = read_big_endian_short((const uint8_t*) tabl e); | 323 uint16_t value = read_big_endian_short((const uint8_t*) tabl e); |
324 gammas[i].fValue = value / 256.0f; | 324 gammas[i].fValue = value / 256.0f; |
325 SkColorSpacePrintf("gamma %d %g\n", value, *gamma); | 325 SkColorSpacePrintf("gamma %d %g\n", value, gammas[i].fValue) ; |
326 break; | 326 break; |
327 } | 327 } |
328 | 328 |
329 // Fill in the interpolation table. | 329 // Check for frequently occurring curves and use a fast approxim ation. |
330 // FIXME (msarett): | 330 if (1024 == count) { |
331 // We should recognize commonly occurring tables and just set ga mma to 2.2f. | 331 // Sample a few values and see if they match our expectation . A more |
332 // robust solution would be to compare each value in this cu rve against | |
scroggo
2016/05/09 17:39:58
nit: It seems like these comments actually apply t
msarett
2016/05/09 19:21:53
Done.
| |
333 // a 2.2f curve see if we remain below an error threshold. At this time, | |
334 // we haven't seen any images in the wild that make this kin d of | |
335 // calculation necessary. We encounter identical gamma curv es over and | |
336 // over again, but relatively few variations. | |
337 if (0 == read_big_endian_short((const uint8_t*) &table[0]) & & | |
338 3341 == read_big_endian_short((const uint8_t*) &tabl e[256]) && | |
339 14057 == read_big_endian_short((const uint8_t*) &tab le[512]) && | |
340 34318 == read_big_endian_short((const uint8_t*) &tab le[768]) && | |
341 65535 == read_big_endian_short((const uint8_t*) &tab le[1023])) { | |
342 gammas[i].fValue = 2.2f; | |
343 break; | |
344 } | |
345 } else if (26 == count) { | |
346 // Sample a few values and see if they match our expectation . | |
347 if (0 == read_big_endian_short((const uint8_t*) &table[0]) & & | |
348 3062 == read_big_endian_short((const uint8_t*) &tabl e[6]) && | |
349 12824 == read_big_endian_short((const uint8_t*) &tab le[12]) && | |
350 31237 == read_big_endian_short((const uint8_t*) &tab le[18]) && | |
351 65535 == read_big_endian_short((const uint8_t*) &tab le[25])) { | |
352 gammas[i].fValue = 2.2f; | |
353 break; | |
354 } | |
355 } | |
356 | |
357 // Otherwise, fill in the interpolation table. | |
332 gammas[i].fTableSize = count; | 358 gammas[i].fTableSize = count; |
333 gammas[i].fTable = std::unique_ptr<float[]>(new float[count]); | 359 gammas[i].fTable = std::unique_ptr<float[]>(new float[count]); |
334 for (uint32_t j = 0; j < count; j++) { | 360 for (uint32_t j = 0; j < count; j++) { |
335 gammas[i].fTable[j] = | 361 gammas[i].fTable[j] = |
336 (read_big_endian_short((const uint8_t*) &table[j])) / 65535.0f; | 362 (read_big_endian_short((const uint8_t*) &table[j])) / 65535.0f; |
337 } | 363 } |
338 break; | 364 break; |
339 } | 365 } |
340 case kTAG_ParaCurveType: | 366 case kTAG_ParaCurveType: |
341 // Guess 2.2f. | 367 // Guess 2.2f. |
(...skipping 22 matching lines...) Expand all Loading... | |
364 SkColorSpacePrintf("Invalid parametric curve type\n"); | 390 SkColorSpacePrintf("Invalid parametric curve type\n"); |
365 return false; | 391 return false; |
366 } | 392 } |
367 break; | 393 break; |
368 default: | 394 default: |
369 SkColorSpacePrintf("Unsupported gamma tag type %d\n", type); | 395 SkColorSpacePrintf("Unsupported gamma tag type %d\n", type); |
370 return false; | 396 return false; |
371 } | 397 } |
372 | 398 |
373 // Adjust src and len if there is another gamma curve to load. | 399 // Adjust src and len if there is another gamma curve to load. |
374 if (0 != numGammas) { | 400 if (i != numGammas - 1) { |
msarett
2016/05/05 20:11:48
Bug fix :)
scroggo
2016/05/09 17:39:58
Might be worth a separate change?
msarett
2016/05/09 19:21:53
You're right, I should have uploaded separately...
| |
375 // Each curve is padded to 4-byte alignment. | 401 // Each curve is padded to 4-byte alignment. |
376 tagBytes = SkAlign4(tagBytes); | 402 tagBytes = SkAlign4(tagBytes); |
377 if (len < tagBytes) { | 403 if (len < tagBytes) { |
378 return false; | 404 return false; |
379 } | 405 } |
380 | 406 |
381 src += tagBytes; | 407 src += tagBytes; |
382 len -= tagBytes; | 408 len -= tagBytes; |
383 } | 409 } |
384 } | 410 } |
(...skipping 247 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
632 toXYZ)); | 658 toXYZ)); |
633 } | 659 } |
634 | 660 |
635 } | 661 } |
636 default: | 662 default: |
637 break; | 663 break; |
638 } | 664 } |
639 | 665 |
640 return_null("ICC profile contains unsupported colorspace"); | 666 return_null("ICC profile contains unsupported colorspace"); |
641 } | 667 } |
OLD | NEW |