/src/libjpeg-turbo.main/src/jdicc.c
Line | Count | Source (jump to first uncovered line) |
1 | | /* |
2 | | * jdicc.c |
3 | | * |
4 | | * Copyright (C) 1997-1998, Thomas G. Lane, Todd Newman. |
5 | | * Copyright (C) 2017, D. R. Commander. |
6 | | * For conditions of distribution and use, see the accompanying README.ijg |
7 | | * file. |
8 | | * |
9 | | * This file provides code to read International Color Consortium (ICC) device |
10 | | * profiles embedded in JFIF JPEG image files. The ICC has defined a standard |
11 | | * for including such data in JPEG "APP2" markers. The code given here does |
12 | | * not know anything about the internal structure of the ICC profile data; it |
13 | | * just knows how to get the profile data from a JPEG file while reading it. |
14 | | */ |
15 | | |
16 | | #define JPEG_INTERNALS |
17 | | #include "jinclude.h" |
18 | | #include "jpeglib.h" |
19 | | #include "jerror.h" |
20 | | |
21 | | |
22 | 19.6k | #define ICC_MARKER (JPEG_APP0 + 2) /* JPEG marker code for ICC */ |
23 | 20.0k | #define ICC_OVERHEAD_LEN 14 /* size of non-profile data in APP2 */ |
24 | | |
25 | | |
26 | | /* |
27 | | * Handy subroutine to test whether a saved marker is an ICC profile marker. |
28 | | */ |
29 | | |
30 | | LOCAL(boolean) |
31 | | marker_is_icc(jpeg_saved_marker_ptr marker) |
32 | 9.81k | { |
33 | 9.81k | return |
34 | 9.81k | marker->marker == ICC_MARKER && |
35 | 9.81k | marker->data_length >= ICC_OVERHEAD_LEN && |
36 | | /* verify the identifying string */ |
37 | 9.81k | marker->data[0] == 0x49 && |
38 | 9.81k | marker->data[1] == 0x43 && |
39 | 9.81k | marker->data[2] == 0x43 && |
40 | 9.81k | marker->data[3] == 0x5F && |
41 | 9.81k | marker->data[4] == 0x50 && |
42 | 9.81k | marker->data[5] == 0x52 && |
43 | 9.81k | marker->data[6] == 0x4F && |
44 | 9.81k | marker->data[7] == 0x46 && |
45 | 9.81k | marker->data[8] == 0x49 && |
46 | 9.81k | marker->data[9] == 0x4C && |
47 | 9.81k | marker->data[10] == 0x45 && |
48 | 9.81k | marker->data[11] == 0x0; |
49 | 9.81k | } |
50 | | |
51 | | |
52 | | /* |
53 | | * See if there was an ICC profile in the JPEG file being read; if so, |
54 | | * reassemble and return the profile data. |
55 | | * |
56 | | * TRUE is returned if an ICC profile was found, FALSE if not. If TRUE is |
57 | | * returned, *icc_data_ptr is set to point to the returned data, and |
58 | | * *icc_data_len is set to its length. |
59 | | * |
60 | | * IMPORTANT: the data at *icc_data_ptr is allocated with malloc() and must be |
61 | | * freed by the caller with free() when the caller no longer needs it. |
62 | | * (Alternatively, we could write this routine to use the IJG library's memory |
63 | | * allocator, so that the data would be freed implicitly when |
64 | | * jpeg_finish_decompress() is called. But it seems likely that many |
65 | | * applications will prefer to have the data stick around after decompression |
66 | | * finishes.) |
67 | | */ |
68 | | |
69 | | GLOBAL(boolean) |
70 | | jpeg_read_icc_profile(j_decompress_ptr cinfo, JOCTET **icc_data_ptr, |
71 | | unsigned int *icc_data_len) |
72 | 5.80k | { |
73 | 5.80k | jpeg_saved_marker_ptr marker; |
74 | 5.80k | int num_markers = 0; |
75 | 5.80k | int seq_no; |
76 | 5.80k | JOCTET *icc_data; |
77 | 5.80k | unsigned int total_length; |
78 | 1.48M | #define MAX_SEQ_NO 255 /* sufficient since marker numbers are bytes */ |
79 | 5.80k | char marker_present[MAX_SEQ_NO + 1]; /* 1 if marker found */ |
80 | 5.80k | unsigned int data_length[MAX_SEQ_NO + 1]; /* size of profile data in marker */ |
81 | 5.80k | unsigned int data_offset[MAX_SEQ_NO + 1]; /* offset for data in marker */ |
82 | | |
83 | 5.80k | if (icc_data_ptr == NULL || icc_data_len == NULL) |
84 | 0 | ERREXIT(cinfo, JERR_BUFFER_SIZE); |
85 | 5.80k | if (cinfo->global_state < DSTATE_READY) |
86 | 0 | ERREXIT1(cinfo, JERR_BAD_STATE, cinfo->global_state); |
87 | | |
88 | 5.80k | *icc_data_ptr = NULL; /* avoid confusion if FALSE return */ |
89 | 5.80k | *icc_data_len = 0; |
90 | | |
91 | | /* This first pass over the saved markers discovers whether there are |
92 | | * any ICC markers and verifies the consistency of the marker numbering. |
93 | | */ |
94 | | |
95 | 1.48M | for (seq_no = 1; seq_no <= MAX_SEQ_NO; seq_no++) |
96 | 1.47M | marker_present[seq_no] = 0; |
97 | | |
98 | 12.6k | for (marker = cinfo->marker_list; marker != NULL; marker = marker->next) { |
99 | 6.83k | if (marker_is_icc(marker)) { |
100 | 237 | if (num_markers == 0) |
101 | 219 | num_markers = marker->data[13]; |
102 | 18 | else if (num_markers != marker->data[13]) { |
103 | 6 | WARNMS(cinfo, JWRN_BOGUS_ICC); /* inconsistent num_markers fields */ |
104 | 6 | return FALSE; |
105 | 6 | } |
106 | 231 | seq_no = marker->data[12]; |
107 | 231 | if (seq_no <= 0 || seq_no > num_markers) { |
108 | 7 | WARNMS(cinfo, JWRN_BOGUS_ICC); /* bogus sequence number */ |
109 | 7 | return FALSE; |
110 | 7 | } |
111 | 224 | if (marker_present[seq_no]) { |
112 | 2 | WARNMS(cinfo, JWRN_BOGUS_ICC); /* duplicate sequence numbers */ |
113 | 2 | return FALSE; |
114 | 2 | } |
115 | 222 | marker_present[seq_no] = 1; |
116 | 222 | data_length[seq_no] = marker->data_length - ICC_OVERHEAD_LEN; |
117 | 222 | } |
118 | 6.83k | } |
119 | | |
120 | 5.78k | if (num_markers == 0) |
121 | 5.58k | return FALSE; |
122 | | |
123 | | /* Check for missing markers, count total space needed, |
124 | | * compute offset of each marker's part of the data. |
125 | | */ |
126 | | |
127 | 204 | total_length = 0; |
128 | 403 | for (seq_no = 1; seq_no <= num_markers; seq_no++) { |
129 | 216 | if (marker_present[seq_no] == 0) { |
130 | 17 | WARNMS(cinfo, JWRN_BOGUS_ICC); /* missing sequence number */ |
131 | 17 | return FALSE; |
132 | 17 | } |
133 | 199 | data_offset[seq_no] = total_length; |
134 | 199 | total_length += data_length[seq_no]; |
135 | 199 | } |
136 | | |
137 | 187 | if (total_length == 0) { |
138 | 1 | WARNMS(cinfo, JWRN_BOGUS_ICC); /* found only empty markers? */ |
139 | 1 | return FALSE; |
140 | 1 | } |
141 | | |
142 | | /* Allocate space for assembled data */ |
143 | 186 | icc_data = (JOCTET *)malloc(total_length * sizeof(JOCTET)); |
144 | 186 | if (icc_data == NULL) |
145 | 0 | ERREXIT1(cinfo, JERR_OUT_OF_MEMORY, 11); /* oops, out of memory */ |
146 | | |
147 | | /* and fill it in */ |
148 | 3.16k | for (marker = cinfo->marker_list; marker != NULL; marker = marker->next) { |
149 | 2.97k | if (marker_is_icc(marker)) { |
150 | 186 | JOCTET FAR *src_ptr; |
151 | 186 | JOCTET *dst_ptr; |
152 | 186 | unsigned int length; |
153 | 186 | seq_no = marker->data[12]; |
154 | 186 | dst_ptr = icc_data + data_offset[seq_no]; |
155 | 186 | src_ptr = marker->data + ICC_OVERHEAD_LEN; |
156 | 186 | length = data_length[seq_no]; |
157 | 10.9k | while (length--) { |
158 | 10.7k | *dst_ptr++ = *src_ptr++; |
159 | 10.7k | } |
160 | 186 | } |
161 | 2.97k | } |
162 | | |
163 | 186 | *icc_data_ptr = icc_data; |
164 | 186 | *icc_data_len = total_length; |
165 | | |
166 | 186 | return TRUE; |
167 | 187 | } |