-
Bug
-
Resolution: Fixed
-
P4
-
7, 8, 9, 11, 12, 13, 14
-
b25
-
generic
-
windows_7
Issue | Fix Version | Assignee | Priority | Status | Resolution | Resolved In Build |
---|---|---|---|---|---|---|
JDK-8246327 | 13.0.4 | Philip Race | P4 | Resolved | Fixed | b04 |
JDK-8241397 | 11.0.8-oracle | Philip Race | P4 | Resolved | Fixed | b01 |
JDK-8241738 | 11.0.8 | Philip Race | P4 | Resolved | Fixed | b01 |
FULL PRODUCT VERSION :
java version "9-ea"
Java(TM) SE Runtime Environment (build 9-ea+157)
Java HotSpot(TM) 64-Bit Server VM (build 9-ea+157, mixed mode)
ADDITIONAL OS VERSION INFORMATION :
Microsoft Windows [Version 6.1.7601]
A DESCRIPTION OF THE PROBLEM :
The code is not only very slow, it has a huge memory footprint that has no relation to the -Xmx setting.
While the code may look silly, it is reduced from a production case where it is done like this because
- we need to initialize the colorspace due to avoid concurrency problems in java (seeJDK-8058973) and to catch other ICC exceptions early (e.g. https://issues.apache.org/jira/browse/PDFBOX-3610 , https://issues.apache.org/jira/browse/PDFBOX-3549 , https://issues.apache.org/jira/browse/PDFBOX-3531 )
- we had a case with 5000 images with separate ICC profile. This took very long and used up to 5 GB
The problem happens also in JDK8 121.
STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
Run
java -Xmx200m -jar PDFBOX3641IccTest.jar
and start the windows task manager and enable the column that shows the used memory. The used memory goes well over 2GB. This code runs for about 7min, and ends with
7 min 15 sec
used memory: 47 MB, max: 209 MB
(I also tried to add the option -XX:MaxMetaspaceSize=200m but this didn't help either)
The values "used memory: 47 MB, max: 209 MB" are similar to what I see in Netbeans profiler telemetry. There used memory never goes very high, but the windows memory usage does.
EXPECTED VERSUS ACTUAL BEHAVIOR :
EXPECTED -
I'd expect that the used memory is well below 1GB.
ACTUAL -
Used memory well over 2GB.
REPRODUCIBILITY :
This bug can be reproduced always.
---------- BEGIN SOURCE ----------
import java.awt.Color;
import java.awt.color.ICC_ColorSpace;
import java.awt.color.ICC_Profile;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
public class PDFBOX3641IccTest
{
public static void main(String[] args) throws IOException
{
ByteArrayOutputStream baos = new ByteArrayOutputStream();
try (InputStream is = new URL("https://issues.apache.org/jira/secure/attachment/12854386/PDFBOX-3641.icc").openStream())
{
int by;
while ((by = is.read()) != -1)
{
baos.write(by);
}
}
byte[] ba = baos.toByteArray();
long t0 = System.currentTimeMillis();
for (int i = 0; i < 5000; ++i)
{
ICC_Profile profile = ICC_Profile.getInstance(ba);
ICC_ColorSpace cs = new ICC_ColorSpace(profile);
cs.toRGB(new float[cs.getNumComponents()]);
cs.fromRGB(new float[3]);
cs.toCIEXYZ(new float[cs.getNumComponents()]);
cs.fromCIEXYZ(new float[3]);
new Color(cs, new float[cs.getNumComponents()], 1f);
// if ((i % 500) == 0)
// Runtime.getRuntime().gc();
}
long duration = (System.currentTimeMillis() - t0) / 1000;
System.out.println(String.format("%d min %d sec", duration / 60, duration % 60));
System.out.println("used memory: " + (Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory()) / 1000000 + " MB, max: " +
Runtime.getRuntime().maxMemory() / 1000000 + " MB");
}
}
---------- END SOURCE ----------
CUSTOMER SUBMITTED WORKAROUND :
Surprisingly, the total used windows memory is much less if the commented-out code is activated, which does a gc() call every 500 iterations. This is kindof weird as I'd expect that gc is "smart" enough to act when memory gets low.
Or could it be that the memory used by ICC is not measured, so that gc doesn't notice by itself that a lot is used?
java version "9-ea"
Java(TM) SE Runtime Environment (build 9-ea+157)
Java HotSpot(TM) 64-Bit Server VM (build 9-ea+157, mixed mode)
ADDITIONAL OS VERSION INFORMATION :
Microsoft Windows [Version 6.1.7601]
A DESCRIPTION OF THE PROBLEM :
The code is not only very slow, it has a huge memory footprint that has no relation to the -Xmx setting.
While the code may look silly, it is reduced from a production case where it is done like this because
- we need to initialize the colorspace due to avoid concurrency problems in java (see
- we had a case with 5000 images with separate ICC profile. This took very long and used up to 5 GB
The problem happens also in JDK8 121.
STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
Run
java -Xmx200m -jar PDFBOX3641IccTest.jar
and start the windows task manager and enable the column that shows the used memory. The used memory goes well over 2GB. This code runs for about 7min, and ends with
7 min 15 sec
used memory: 47 MB, max: 209 MB
(I also tried to add the option -XX:MaxMetaspaceSize=200m but this didn't help either)
The values "used memory: 47 MB, max: 209 MB" are similar to what I see in Netbeans profiler telemetry. There used memory never goes very high, but the windows memory usage does.
EXPECTED VERSUS ACTUAL BEHAVIOR :
EXPECTED -
I'd expect that the used memory is well below 1GB.
ACTUAL -
Used memory well over 2GB.
REPRODUCIBILITY :
This bug can be reproduced always.
---------- BEGIN SOURCE ----------
import java.awt.Color;
import java.awt.color.ICC_ColorSpace;
import java.awt.color.ICC_Profile;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
public class PDFBOX3641IccTest
{
public static void main(String[] args) throws IOException
{
ByteArrayOutputStream baos = new ByteArrayOutputStream();
try (InputStream is = new URL("https://issues.apache.org/jira/secure/attachment/12854386/PDFBOX-3641.icc").openStream())
{
int by;
while ((by = is.read()) != -1)
{
baos.write(by);
}
}
byte[] ba = baos.toByteArray();
long t0 = System.currentTimeMillis();
for (int i = 0; i < 5000; ++i)
{
ICC_Profile profile = ICC_Profile.getInstance(ba);
ICC_ColorSpace cs = new ICC_ColorSpace(profile);
cs.toRGB(new float[cs.getNumComponents()]);
cs.fromRGB(new float[3]);
cs.toCIEXYZ(new float[cs.getNumComponents()]);
cs.fromCIEXYZ(new float[3]);
new Color(cs, new float[cs.getNumComponents()], 1f);
// if ((i % 500) == 0)
// Runtime.getRuntime().gc();
}
long duration = (System.currentTimeMillis() - t0) / 1000;
System.out.println(String.format("%d min %d sec", duration / 60, duration % 60));
System.out.println("used memory: " + (Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory()) / 1000000 + " MB, max: " +
Runtime.getRuntime().maxMemory() / 1000000 + " MB");
}
}
---------- END SOURCE ----------
CUSTOMER SUBMITTED WORKAROUND :
Surprisingly, the total used windows memory is much less if the commented-out code is activated, which does a gc() call every 500 iterations. This is kindof weird as I'd expect that gc is "smart" enough to act when memory gets low.
Or could it be that the memory used by ICC is not measured, so that gc doesn't notice by itself that a lot is used?
- backported by
-
JDK-8241397 ICC_Profile has un-needed, not-empty finalize method
- Resolved
-
JDK-8241738 ICC_Profile has un-needed, not-empty finalize method
- Resolved
-
JDK-8246327 ICC_Profile has un-needed, not-empty finalize method
- Resolved