Skip to content

Commit ed8945a

Browse files
author
Harshitha Onkar
committedFeb 6, 2025
8347377: Add validation checks for ICC_Profile header fields
Reviewed-by: prr, jdv
1 parent 1ab1c1d commit ed8945a

File tree

4 files changed

+327
-2
lines changed

4 files changed

+327
-2
lines changed
 

‎src/java.desktop/share/classes/java/awt/color/ICC_ColorSpace.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -141,10 +141,11 @@ public ICC_ColorSpace(ICC_Profile profile) {
141141
if (profileClass != ICC_Profile.CLASS_INPUT
142142
&& profileClass != ICC_Profile.CLASS_DISPLAY
143143
&& profileClass != ICC_Profile.CLASS_OUTPUT
144+
&& profileClass != ICC_Profile.CLASS_DEVICELINK
144145
&& profileClass != ICC_Profile.CLASS_COLORSPACECONVERSION
145146
&& profileClass != ICC_Profile.CLASS_NAMEDCOLOR
146147
&& profileClass != ICC_Profile.CLASS_ABSTRACT) {
147-
throw new IllegalArgumentException("Invalid profile type");
148+
throw new IllegalArgumentException("Invalid profile class");
148149
}
149150

150151
thisProfile = profile;

‎src/java.desktop/share/classes/java/awt/color/ICC_Profile.java

+64-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1997, 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1997, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -756,6 +756,7 @@ private interface BuiltInProfile {
756756
*/
757757
public static final int icXYZNumberX = 8;
758758

759+
private static final int HEADER_SIZE = 128;
759760

760761
/**
761762
* Constructs an {@code ICC_Profile} object with a given ID.
@@ -786,10 +787,15 @@ public static ICC_Profile getInstance(byte[] data) {
786787
ProfileDataVerifier.verify(data);
787788
Profile p;
788789
try {
790+
byte[] theHeader = new byte[HEADER_SIZE];
791+
System.arraycopy(data, 0, theHeader, 0, HEADER_SIZE);
792+
verifyHeader(theHeader);
793+
789794
p = CMSManager.getModule().loadProfile(data);
790795
} catch (CMMException c) {
791796
throw new IllegalArgumentException("Invalid ICC Profile Data");
792797
}
798+
793799
try {
794800
if (getColorSpaceType(p) == ColorSpace.TYPE_GRAY
795801
&& getData(p, icSigMediaWhitePointTag) != null
@@ -972,6 +978,10 @@ public int getProfileClass() {
972978
return info.profileClass;
973979
}
974980
byte[] theHeader = getData(icSigHead);
981+
return getProfileClass(theHeader);
982+
}
983+
984+
private static int getProfileClass(byte[] theHeader) {
975985
int theClassSig = intFromBigEndian(theHeader, icHdrDeviceClass);
976986
return switch (theClassSig) {
977987
case icSigInputClass -> CLASS_INPUT;
@@ -1013,6 +1023,11 @@ private static int getColorSpaceType(Profile p) {
10131023
return iccCStoJCS(theColorSpaceSig);
10141024
}
10151025

1026+
private static int getColorSpaceType(byte[] theHeader) {
1027+
int theColorSpaceSig = intFromBigEndian(theHeader, icHdrColorSpace);
1028+
return iccCStoJCS(theColorSpaceSig);
1029+
}
1030+
10161031
/**
10171032
* Returns the color space type of the Profile Connection Space (PCS).
10181033
* Returns one of the color space type constants defined by the ColorSpace
@@ -1032,6 +1047,21 @@ public int getPCSType() {
10321047
return iccCStoJCS(thePCSSig);
10331048
}
10341049

1050+
private static int getPCSType(byte[] theHeader) {
1051+
int thePCSSig = intFromBigEndian(theHeader, icHdrPcs);
1052+
int theDeviceClass = intFromBigEndian(theHeader, icHdrDeviceClass);
1053+
1054+
if (theDeviceClass == icSigLinkClass) {
1055+
return iccCStoJCS(thePCSSig);
1056+
} else {
1057+
return switch (thePCSSig) {
1058+
case icSigXYZData -> ColorSpace.TYPE_XYZ;
1059+
case icSigLabData -> ColorSpace.TYPE_Lab;
1060+
default -> throw new IllegalArgumentException("Unexpected PCS type");
1061+
};
1062+
}
1063+
}
1064+
10351065
/**
10361066
* Write this {@code ICC_Profile} to a file.
10371067
*
@@ -1112,9 +1142,42 @@ private static byte[] getData(Profile p, int tagSignature) {
11121142
* @see #getData
11131143
*/
11141144
public void setData(int tagSignature, byte[] tagData) {
1145+
if (tagSignature == ICC_Profile.icSigHead) {
1146+
verifyHeader(tagData);
1147+
}
11151148
CMSManager.getModule().setTagData(cmmProfile(), tagSignature, tagData);
11161149
}
11171150

1151+
private static void verifyHeader(byte[] data) {
1152+
if (data == null || data.length < HEADER_SIZE) {
1153+
throw new IllegalArgumentException("Invalid header data");
1154+
}
1155+
getProfileClass(data);
1156+
getColorSpaceType(data);
1157+
getPCSType(data);
1158+
checkRenderingIntent(data);
1159+
}
1160+
1161+
private static boolean checkRenderingIntent(byte[] header) {
1162+
int index = ICC_Profile.icHdrRenderingIntent;
1163+
1164+
/* According to ICC spec, only the least-significant 16 bits shall be
1165+
* used to encode the rendering intent. The most significant 16 bits
1166+
* shall be set to zero. Thus, we are ignoring two most significant
1167+
* bytes here. Please refer ICC Spec Document for more details.
1168+
*/
1169+
int renderingIntent = ((header[index+2] & 0xff) << 8) |
1170+
(header[index+3] & 0xff);
1171+
1172+
switch (renderingIntent) {
1173+
case icPerceptual, icMediaRelativeColorimetric,
1174+
icSaturation, icAbsoluteColorimetric -> {
1175+
return true;
1176+
}
1177+
default -> throw new IllegalArgumentException("Unknown Rendering Intent");
1178+
}
1179+
}
1180+
11181181
/**
11191182
* Returns the number of color components in the "input" color space of this
11201183
* profile. For example if the color space type of this profile is
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,261 @@
1+
/*
2+
* Copyright (c) 2025, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test
26+
* @bug 8337703
27+
* @summary To verify if ICC_Profile's setData() and getInstance() methods
28+
* validate header data and throw IAE for invalid values.
29+
* @run main ValidateICCHeaderData
30+
*/
31+
32+
import java.awt.color.ColorSpace;
33+
import java.awt.color.ICC_Profile;
34+
import java.awt.image.BufferedImage;
35+
import java.io.IOException;
36+
import java.math.BigInteger;
37+
import java.nio.ByteBuffer;
38+
39+
public class ValidateICCHeaderData {
40+
private static ICC_Profile profile;
41+
42+
private static final boolean DEBUG = false;
43+
private static final int VALID_HEADER_SIZE = 128;
44+
private static final int HEADER_TAG = ICC_Profile.icSigHead;
45+
private static final int PROFILE_CLASS_START_INDEX = ICC_Profile.icHdrDeviceClass;
46+
private static final int COLOR_SPACE_START_INDEX = ICC_Profile.icHdrColorSpace;
47+
private static final int RENDER_INTENT_START_INDEX = ICC_Profile.icHdrRenderingIntent;
48+
private static final int PCS_START_INDEX = ICC_Profile.icHdrPcs;
49+
50+
private static final int[] VALID_PROFILE_CLASS = new int[] {
51+
ICC_Profile.icSigInputClass, ICC_Profile.icSigDisplayClass,
52+
ICC_Profile.icSigOutputClass, ICC_Profile.icSigLinkClass,
53+
ICC_Profile.icSigAbstractClass, ICC_Profile.icSigColorSpaceClass,
54+
ICC_Profile.icSigNamedColorClass
55+
};
56+
57+
private static final int[] VALID_COLOR_SPACE = new int[] {
58+
ICC_Profile.icSigXYZData, ICC_Profile.icSigLabData,
59+
ICC_Profile.icSigLuvData, ICC_Profile.icSigYCbCrData,
60+
ICC_Profile.icSigYxyData, ICC_Profile.icSigRgbData,
61+
ICC_Profile.icSigGrayData, ICC_Profile.icSigHsvData,
62+
ICC_Profile.icSigHlsData, ICC_Profile.icSigCmykData,
63+
ICC_Profile.icSigSpace2CLR, ICC_Profile.icSigSpace3CLR,
64+
ICC_Profile.icSigSpace4CLR, ICC_Profile.icSigSpace5CLR,
65+
ICC_Profile.icSigSpace6CLR, ICC_Profile.icSigSpace7CLR,
66+
ICC_Profile.icSigSpace8CLR, ICC_Profile.icSigSpace9CLR,
67+
ICC_Profile.icSigSpaceACLR, ICC_Profile.icSigSpaceBCLR,
68+
ICC_Profile.icSigSpaceCCLR, ICC_Profile.icSigSpaceDCLR,
69+
ICC_Profile.icSigSpaceECLR, ICC_Profile.icSigSpaceFCLR,
70+
ICC_Profile.icSigCmyData
71+
};
72+
73+
private static final int[] VALID_RENDER_INTENT = new int[] {
74+
ICC_Profile.icPerceptual, ICC_Profile.icMediaRelativeColorimetric,
75+
ICC_Profile.icSaturation, ICC_Profile.icAbsoluteColorimetric
76+
};
77+
78+
private static void createCopyOfBuiltInProfile() {
79+
ICC_Profile builtInProfile = ICC_Profile.getInstance(ColorSpace.CS_sRGB);
80+
//copy of SRGB BuiltIn Profile that can be modified
81+
//using ICC_Profile.setData()
82+
profile = ICC_Profile.getInstance(builtInProfile.getData());
83+
}
84+
85+
public static void main(String[] args) throws Exception {
86+
createCopyOfBuiltInProfile();
87+
88+
System.out.println("CASE 1: Testing VALID Profile Classes ...");
89+
testValidHeaderData(VALID_PROFILE_CLASS, PROFILE_CLASS_START_INDEX, 4);
90+
System.out.println("CASE 1: Passed \n");
91+
92+
// PCS field validation for Profile class != DEVICE_LINK
93+
System.out.println("CASE 2: Testing VALID PCS Type"
94+
+ " for Profile class != DEVICE_LINK ...");
95+
testValidHeaderData(new int[] {ICC_Profile.icSigXYZData, ICC_Profile.icSigLabData},
96+
PCS_START_INDEX, 4);
97+
System.out.println("CASE 2: Passed \n");
98+
99+
System.out.println("CASE 3: Testing INVALID PCS Type"
100+
+ " for Profile class != DEVICE_LINK ...");
101+
testInvalidHeaderData(ICC_Profile.icSigCmykData, PCS_START_INDEX, 4);
102+
System.out.println("CASE 3: Passed \n");
103+
104+
System.out.println("CASE 4: Testing DEVICE LINK PROFILE CLASS ...");
105+
testValidHeaderData(new int[] {ICC_Profile.icSigLinkClass},
106+
PROFILE_CLASS_START_INDEX, 4);
107+
//to check if instantiating BufferedImage with
108+
//ICC_Profile device class = CLASS_DEVICELINK does not throw IAE.
109+
BufferedImage img = new BufferedImage(100, 100,
110+
BufferedImage.TYPE_3BYTE_BGR);
111+
System.out.println("CASE 4: Passed \n");
112+
113+
// PCS field validation for Profile class == DEVICE_LINK
114+
System.out.println("CASE 5: Testing VALID PCS Type"
115+
+ " for Profile class == DEVICE_LINK ...");
116+
testValidHeaderData(VALID_COLOR_SPACE, PCS_START_INDEX, 4);
117+
System.out.println("CASE 5: Passed \n");
118+
119+
System.out.println("CASE 6: Testing INVALID PCS Type"
120+
+ " for Profile class == DEVICE_LINK ...");
121+
//original icSigLabData = 0x4C616220
122+
int invalidSigLabData = 0x4C616221;
123+
testInvalidHeaderData(invalidSigLabData, PCS_START_INDEX, 4);
124+
System.out.println("CASE 6: Passed \n");
125+
126+
System.out.println("CASE 7: Testing VALID Color Spaces ...");
127+
testValidHeaderData(VALID_COLOR_SPACE, COLOR_SPACE_START_INDEX, 4);
128+
System.out.println("CASE 7: Passed \n");
129+
130+
System.out.println("CASE 8: Testing VALID Rendering Intent ...");
131+
testValidHeaderData(VALID_RENDER_INTENT, RENDER_INTENT_START_INDEX, 4);
132+
System.out.println("CASE 8: Passed \n");
133+
134+
System.out.println("CASE 9: Testing INVALID Profile Class ...");
135+
//original icSigInputClass = 0x73636E72
136+
int invalidSigInputClass = 0x73636E70;
137+
testInvalidHeaderData(invalidSigInputClass, PROFILE_CLASS_START_INDEX, 4);
138+
System.out.println("CASE 9: Passed \n");
139+
140+
System.out.println("CASE 10: Testing INVALID Color Space ...");
141+
//original icSigXYZData = 0x58595A20
142+
int invalidSigXYZData = 0x58595A21;
143+
testInvalidHeaderData(invalidSigXYZData, COLOR_SPACE_START_INDEX, 4);
144+
System.out.println("CASE 10: Passed \n");
145+
146+
System.out.println("CASE 11: Testing INVALID Rendering Intent ...");
147+
//valid rendering intent values are 0-3
148+
int invalidRenderIntent = 5;
149+
testInvalidHeaderData(invalidRenderIntent, RENDER_INTENT_START_INDEX, 4);
150+
System.out.println("CASE 11: Passed \n");
151+
152+
System.out.println("CASE 12: Testing INVALID Header Size ...");
153+
testInvalidHeaderSize();
154+
System.out.println("CASE 12: Passed \n");
155+
156+
System.out.println("CASE 13: Testing ICC_Profile.getInstance(..)"
157+
+ " with VALID profile data ...");
158+
testProfileCreation(true);
159+
System.out.println("CASE 13: Passed \n");
160+
161+
System.out.println("CASE 14: Testing ICC_Profile.getInstance(..)"
162+
+ " with INVALID profile data ...");
163+
testProfileCreation(false);
164+
System.out.println("CASE 14: Passed \n");
165+
166+
System.out.println("CASE 15: Testing Deserialization of ICC_Profile ...");
167+
testDeserialization();
168+
System.out.println("CASE 15: Passed \n");
169+
170+
System.out.println("Successfully completed testing all 15 cases. Test Passed !!");
171+
}
172+
173+
private static void testValidHeaderData(int[] validData, int startIndex,
174+
int fieldLength) {
175+
for (int value : validData) {
176+
setTag(value, startIndex, fieldLength);
177+
}
178+
}
179+
180+
private static void testInvalidHeaderData(int invalidData, int startIndex,
181+
int fieldLength) {
182+
try {
183+
setTag(invalidData, startIndex, fieldLength);
184+
throw new RuntimeException("Test Failed ! Expected IAE NOT thrown");
185+
} catch (IllegalArgumentException iae) {
186+
System.out.println("Expected IAE thrown: " + iae.getMessage());
187+
}
188+
}
189+
190+
private static void setTag(int value, int startIndex, int fieldLength) {
191+
byte[] byteArray;
192+
if (startIndex == RENDER_INTENT_START_INDEX) {
193+
byteArray = ByteBuffer.allocate(4).putInt(value).array();
194+
} else {
195+
BigInteger big = BigInteger.valueOf(value);
196+
byteArray = (big.toByteArray());
197+
}
198+
199+
if (DEBUG) {
200+
System.out.print("Byte Array : ");
201+
for (int i = 0; i < byteArray.length; i++) {
202+
System.out.print(byteArray[i] + " ");
203+
}
204+
System.out.println("\n");
205+
}
206+
207+
byte[] iccProfileHeaderData = profile.getData(HEADER_TAG);
208+
System.arraycopy(byteArray, 0, iccProfileHeaderData, startIndex, fieldLength);
209+
profile.setData(HEADER_TAG, iccProfileHeaderData);
210+
}
211+
212+
private static void testProfileCreation(boolean validCase) {
213+
ICC_Profile builtInProfile = ICC_Profile.getInstance(ColorSpace.CS_GRAY);
214+
byte[] profileData = builtInProfile.getData();
215+
216+
int validDeviceClass = ICC_Profile.icSigInputClass;
217+
BigInteger big = BigInteger.valueOf(validDeviceClass);
218+
//valid case set device class to 0x73636E72 (icSigInputClass)
219+
//invalid case set device class to 0x00000000
220+
byte[] field = validCase ? big.toByteArray()
221+
: ByteBuffer.allocate(4).putInt(0).array();
222+
System.arraycopy(field, 0, profileData, PROFILE_CLASS_START_INDEX, 4);
223+
224+
try {
225+
ICC_Profile.getInstance(profileData);
226+
if (!validCase) {
227+
throw new RuntimeException("Test Failed ! Expected IAE NOT thrown");
228+
}
229+
} catch (IllegalArgumentException iae) {
230+
if (!validCase) {
231+
System.out.println("Expected IAE thrown: " + iae.getMessage());
232+
} else {
233+
throw new RuntimeException("Unexpected IAE thrown");
234+
}
235+
}
236+
}
237+
238+
private static void testInvalidHeaderSize() {
239+
byte[] iccProfileHeaderData = profile.getData(HEADER_TAG);
240+
byte[] invalidHeaderSize = new byte[VALID_HEADER_SIZE - 1];
241+
System.arraycopy(iccProfileHeaderData, 0,
242+
invalidHeaderSize, 0, invalidHeaderSize.length);
243+
try {
244+
profile.setData(HEADER_TAG, invalidHeaderSize);
245+
throw new RuntimeException("Test Failed ! Expected IAE NOT thrown");
246+
} catch (IllegalArgumentException iae) {
247+
System.out.println("Expected IAE thrown: " + iae.getMessage());
248+
}
249+
}
250+
251+
private static void testDeserialization() throws IOException {
252+
//invalidSRGB.icc is serialized on older version of JDK
253+
//Upon deserialization, the invalid profile is expected to throw IAE
254+
try {
255+
ICC_Profile.getInstance("./invalidSRGB.icc");
256+
throw new RuntimeException("Test Failed ! Expected IAE NOT thrown");
257+
} catch (IllegalArgumentException iae) {
258+
System.out.println("Expected IAE thrown: " + iae.getMessage());
259+
}
260+
}
261+
}
Binary file not shown.

0 commit comments

Comments
 (0)
Please sign in to comment.