Backport to fix iTXt chunk handling for pngs

Deepak Bhole dbhole at redhat.com
Thu Dec 23 12:38:31 PST 2010


Hi,

Currently, iTXt chunks in pngs are handled incorrectly in icedtea6. This
has been reported as 6541476 in the Oracle bug db and has been fixed in
OpenJDK7 for a while:
http://hg.openjdk.java.net/jdk7/jdk7/jdk/rev/3a9d06af8830

In order to cleanly apply the above, this patch needs to be applied as
well:
http://hg.openjdk.java.net/jdk7/jdk7/jdk/rev/94bcd3503668

94bcd3503668 is already in the jdk6 repo and will be in b21. However, 
3a9d06af8830 isn't in jdk6 and I've requested that it be backported[1].

Since icedtea6 is currently at b20, both of the above commits need to
be applied. Attached patch does so. There is also an rhbz associated
with this issue:
https://bugzilla.redhat.com/show_bug.cgi?id=665355

Given that both have been in 7 for over 2 years, I'd like to propose
that this also be applied to 1.7, 1.8 and 1.9.

ChangeLog:

-------------------------------------------------------------------------------
2010-12-23  Deepak Bhole <dbhole at redhat.com>

   Backport S6687968, S6541476
   * Makefile.am: Add missing "\" to the patch list, update to include 
   new patches.
   * NEWS: Updated.
   * patches/openjdk/6687968-pngimagereader_mem_leak.patch: Backport to allow
   S6541476 fix to apply cleanly.
   * patches/openjdk/6541476-png-iTXt-chunk.patch: Backport to fix iTXt chunk
   handling for png images. Also addresses RH665355.
-------------------------------------------------------------------------------

Cheers,
Deepak

[1]: Request for JDK6 backport of 3a9d06af8830:
http://mail.openjdk.java.net/pipermail/jdk6-dev/2010-December/002182.html


-------------- next part --------------
diff -r 7219bd74962a Makefile.am
--- a/Makefile.am	Thu Dec 23 14:56:09 2010 +0100
+++ b/Makefile.am	Thu Dec 23 15:20:57 2010 -0500
@@ -312,8 +312,10 @@
 	patches/openjdk/6979979-gtk_font_size_rounding.patch \
 	patches/openjdk/6795356-proxylazyvalue-leak.patch \
 	patches/rendering-engine-tests.patch \
-	patches/jtreg-TestXEmbedServer-fix.patch
-	patches/openjdk/6800846-printing-quality.patch
+	patches/jtreg-TestXEmbedServer-fix.patch \
+	patches/openjdk/6800846-printing-quality.patch \
+	patches/openjdk/6687968-pngimagereader_mem_leak.patch \
+	patches/openjdk/6541476-png-iTXt-chunk.patch
 
 if WITH_ALT_HSBUILD
 ICEDTEA_PATCHES += \
diff -r 7219bd74962a NEWS
--- a/NEWS	Thu Dec 23 14:56:09 2010 +0100
+++ b/NEWS	Thu Dec 23 15:20:57 2010 -0500
@@ -51,6 +51,8 @@
   - S6979979, RH508185: Rounding error in font sizes selected by the GTK Look and Feel
   - S6795356: Leak caused by javax.swing.UIDefaults.ProxyLazyValue.acc
   - S6800846, RH662230: Printing quality degraded with Java 6 compared to 5.0, index out of bounds exception.
+  - S6687968: PNGImageReader leaks native memory through an Inflater
+  - S6541476, RH665355: PNG imageio plugin incorrectly handles iTXt chunk
 
 * Bug fixes
   - S7003777, RH647674: JTextPane produces incorrect content after parsing the html text
diff -r 7219bd74962a patches/openjdk/6541476-png-iTXt-chunk.patch
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/patches/openjdk/6541476-png-iTXt-chunk.patch	Thu Dec 23 15:20:57 2010 -0500
@@ -0,0 +1,549 @@
+diff -urN openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageReader.java openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageReader.java
+--- openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageReader.java	2010-12-23 15:16:15.107446365 -0500
++++ openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageReader.java	2010-12-23 15:17:00.433446450 -0500
+@@ -44,7 +44,6 @@
+ import java.util.Arrays;
+ import java.util.Enumeration;
+ import java.util.Iterator;
+-import java.util.List;
+ import java.util.zip.Inflater;
+ import java.util.zip.InflaterInputStream;
+ import javax.imageio.IIOException;
+@@ -57,6 +56,7 @@
+ import com.sun.imageio.plugins.common.InputStreamAdapter;
+ import com.sun.imageio.plugins.common.ReaderUtil;
+ import com.sun.imageio.plugins.common.SubImageInputStream;
++import java.io.ByteArrayOutputStream;
+ import sun.awt.image.ByteInterleavedRaster;
+ 
+ class PNGImageDataEnumeration implements Enumeration {
+@@ -209,6 +209,15 @@
+         resetStreamSettings();
+     }
+ 
++    private String readNullTerminatedString(String charset) throws IOException {
++        ByteArrayOutputStream baos = new ByteArrayOutputStream();
++        int b;
++        while ((b = stream.read()) != 0) {
++            baos.write(b);
++        }
++        return new String(baos.toByteArray(), charset);
++    }
++
+     private String readNullTerminatedString() throws IOException {
+         StringBuilder b = new StringBuilder();
+         int c;
+@@ -447,26 +456,27 @@
+         metadata.iTXt_keyword.add(keyword);
+ 
+         int compressionFlag = stream.readUnsignedByte();
+-        metadata.iTXt_compressionFlag.add(new Integer(compressionFlag));
++        metadata.iTXt_compressionFlag.add(Boolean.valueOf(compressionFlag == 1));
+ 
+         int compressionMethod = stream.readUnsignedByte();
+-        metadata.iTXt_compressionMethod.add(new Integer(compressionMethod));
++        metadata.iTXt_compressionMethod.add(Integer.valueOf(compressionMethod));
+ 
+-        String languageTag = readNullTerminatedString();
++        String languageTag = readNullTerminatedString("UTF8");
+         metadata.iTXt_languageTag.add(languageTag);
+ 
+-        String translatedKeyword = stream.readUTF();
++        String translatedKeyword =
++            readNullTerminatedString("UTF8");
+         metadata.iTXt_translatedKeyword.add(translatedKeyword);
+-        stream.skipBytes(1); // Null separator
+ 
+         String text;
++        long pos = stream.getStreamPosition();
++        byte[] b = new byte[(int)(chunkStart + chunkLength - pos)];
++        stream.readFully(b);
++
+         if (compressionFlag == 1) { // Decompress the text
+-            long pos = stream.getStreamPosition();
+-            byte[] b = new byte[(int)(chunkStart + chunkLength - pos)];
+-            stream.readFully(b);
+-            text = inflate(b);
++            text = new String(inflate(b), "UTF8");
+         } else {
+-            text = stream.readUTF();
++            text = new String(b, "UTF8");
+         }
+         metadata.iTXt_text.add(text);
+     }
+@@ -615,20 +625,20 @@
+         metadata.tRNS_present = true;
+     }
+ 
+-    private static String inflate(byte[] b) throws IOException {
++    private static byte[] inflate(byte[] b) throws IOException {
+         InputStream bais = new ByteArrayInputStream(b);
+         InputStream iis = new InflaterInputStream(bais);
++        ByteArrayOutputStream baos = new ByteArrayOutputStream();
+ 
+-        StringBuilder sb = new StringBuilder(80);
+         int c;
+         try {
+             while ((c = iis.read()) != -1) {
+-                sb.append((char)c);
++                baos.write(c);
+             }
+         } finally {
+             iis.close();
+         }
+-        return sb.toString();
++        return baos.toByteArray();
+     }
+ 
+     private void parse_zTXt_chunk(int chunkLength) throws IOException {
+@@ -640,7 +650,7 @@
+ 
+         byte[] b = new byte[chunkLength - keyword.length() - 2];
+         stream.readFully(b);
+-        metadata.zTXt_text.add(inflate(b));
++        metadata.zTXt_text.add(new String(inflate(b)));
+     }
+ 
+     private void readMetadata() throws IIOException {
+diff -urN openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageWriter.java openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageWriter.java
+--- openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageWriter.java	2010-12-23 15:16:15.107446365 -0500
++++ openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageWriter.java	2010-12-23 15:17:00.434446439 -0500
+@@ -671,13 +671,13 @@
+         }
+     }
+ 
+-    private byte[] deflate(String s) throws IOException {
++    private byte[] deflate(byte[] b) throws IOException {
+         ByteArrayOutputStream baos = new ByteArrayOutputStream();
+         DeflaterOutputStream dos = new DeflaterOutputStream(baos);
+ 
+-        int len = s.length();
++        int len = b.length;
+         for (int i = 0; i < len; i++) {
+-            dos.write((int)s.charAt(i));
++            dos.write((int)(0xff & b[i]));
+         }
+         dos.close();
+ 
+@@ -685,38 +685,37 @@
+     }
+ 
+     private void write_iTXt() throws IOException {
+-        Iterator keywordIter = metadata.iTXt_keyword.iterator();
+-        Iterator flagIter = metadata.iTXt_compressionFlag.iterator();
+-        Iterator methodIter = metadata.iTXt_compressionMethod.iterator();
+-        Iterator languageIter = metadata.iTXt_languageTag.iterator();
+-        Iterator translatedKeywordIter =
++        Iterator<String> keywordIter = metadata.iTXt_keyword.iterator();
++        Iterator<Boolean> flagIter = metadata.iTXt_compressionFlag.iterator();
++        Iterator<Integer> methodIter = metadata.iTXt_compressionMethod.iterator();
++        Iterator<String> languageIter = metadata.iTXt_languageTag.iterator();
++        Iterator<String> translatedKeywordIter =
+             metadata.iTXt_translatedKeyword.iterator();
+-        Iterator textIter = metadata.iTXt_text.iterator();
++        Iterator<String> textIter = metadata.iTXt_text.iterator();
+ 
+         while (keywordIter.hasNext()) {
+             ChunkStream cs = new ChunkStream(PNGImageReader.iTXt_TYPE, stream);
+-            String keyword = (String)keywordIter.next();
+-            cs.writeBytes(keyword);
++
++            cs.writeBytes(keywordIter.next());
+             cs.writeByte(0);
+ 
+-            int flag = ((Integer)flagIter.next()).intValue();
+-            cs.writeByte(flag);
+-            int method = ((Integer)methodIter.next()).intValue();
+-            cs.writeByte(method);
++            Boolean compressed = flagIter.next();
++            cs.writeByte(compressed ? 1 : 0);
+ 
+-            String languageTag = (String)languageIter.next();
+-            cs.writeBytes(languageTag);
++            cs.writeByte(methodIter.next().intValue());
++
++            cs.writeBytes(languageIter.next());
+             cs.writeByte(0);
+ 
+-            String translatedKeyword = (String)translatedKeywordIter.next();
+-            cs.writeBytes(translatedKeyword);
++
++            cs.write(translatedKeywordIter.next().getBytes("UTF8"));
+             cs.writeByte(0);
+ 
+-            String text = (String)textIter.next();
+-            if (flag == 1) {
+-                cs.write(deflate(text));
++            String text = textIter.next();
++            if (compressed) {
++                cs.write(deflate(text.getBytes("UTF8")));
+             } else {
+-                cs.writeUTF(text);
++                cs.write(text.getBytes("UTF8"));
+             }
+             cs.finish();
+         }
+@@ -737,7 +736,7 @@
+             cs.writeByte(compressionMethod);
+ 
+             String text = (String)textIter.next();
+-            cs.write(deflate(text));
++            cs.write(deflate(text.getBytes()));
+             cs.finish();
+         }
+     }
+diff -urN openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGMetadata.java openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGMetadata.java
+--- openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGMetadata.java	2010-06-21 17:15:11.000000000 -0400
++++ openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGMetadata.java	2010-12-23 15:17:00.436446415 -0500
+@@ -176,12 +176,12 @@
+     public byte[] iCCP_compressedProfile;
+ 
+     // iTXt chunk
+-    public ArrayList iTXt_keyword = new ArrayList(); // Strings
+-    public ArrayList iTXt_compressionFlag = new ArrayList(); // Integers
+-    public ArrayList iTXt_compressionMethod = new ArrayList(); // Integers
+-    public ArrayList iTXt_languageTag = new ArrayList(); // Strings
+-    public ArrayList iTXt_translatedKeyword = new ArrayList(); // Strings
+-    public ArrayList iTXt_text = new ArrayList(); // Strings
++    public ArrayList<String> iTXt_keyword = new ArrayList<String>();
++    public ArrayList<Boolean> iTXt_compressionFlag = new ArrayList<Boolean>();
++    public ArrayList<Integer> iTXt_compressionMethod = new ArrayList<Integer>();
++    public ArrayList<String> iTXt_languageTag = new ArrayList<String>();
++    public ArrayList<String> iTXt_translatedKeyword = new ArrayList<String>();
++    public ArrayList<String> iTXt_text = new ArrayList<String>();
+ 
+     // pHYs chunk
+     public boolean pHYs_present;
+@@ -599,19 +599,17 @@
+         if (iTXt_keyword.size() > 0) {
+             IIOMetadataNode iTXt_parent = new IIOMetadataNode("iTXt");
+             for (int i = 0; i < iTXt_keyword.size(); i++) {
+-                Integer val;
+-
+                 IIOMetadataNode iTXt_node = new IIOMetadataNode("iTXtEntry");
+-                iTXt_node.setAttribute("keyword", (String)iTXt_keyword.get(i));
+-                val = (Integer)iTXt_compressionFlag.get(i);
+-                iTXt_node.setAttribute("compressionFlag", val.toString());
+-                val = (Integer)iTXt_compressionMethod.get(i);
+-                iTXt_node.setAttribute("compressionMethod", val.toString());
++                iTXt_node.setAttribute("keyword", iTXt_keyword.get(i));
++                iTXt_node.setAttribute("compressionFlag",
++                        iTXt_compressionFlag.get(i) ? "1" : "0");
++                iTXt_node.setAttribute("compressionMethod",
++                        iTXt_compressionMethod.get(i).toString());
+                 iTXt_node.setAttribute("languageTag",
+-                                       (String)iTXt_languageTag.get(i));
++                                       iTXt_languageTag.get(i));
+                 iTXt_node.setAttribute("translatedKeyword",
+-                                       (String)iTXt_translatedKeyword.get(i));
+-                iTXt_node.setAttribute("text", (String)iTXt_text.get(i));
++                                       iTXt_translatedKeyword.get(i));
++                iTXt_node.setAttribute("text", iTXt_text.get(i));
+ 
+                 iTXt_parent.appendChild(iTXt_node);
+             }
+@@ -1039,11 +1037,11 @@
+ 
+         for (int i = 0; i < iTXt_keyword.size(); i++) {
+             node = new IIOMetadataNode("TextEntry");
+-            node.setAttribute("keyword", (String)iTXt_keyword.get(i));
+-            node.setAttribute("value", (String)iTXt_text.get(i));
++            node.setAttribute("keyword", iTXt_keyword.get(i));
++            node.setAttribute("value", iTXt_text.get(i));
+             node.setAttribute("language",
+-                              (String)iTXt_languageTag.get(i));
+-            if (((Integer)iTXt_compressionFlag.get(i)).intValue() == 1) {
++                              iTXt_languageTag.get(i));
++            if (iTXt_compressionFlag.get(i)) {
+                 node.setAttribute("compression", "deflate");
+             } else {
+                 node.setAttribute("compression", "none");
+@@ -1429,11 +1427,11 @@
+ 
+                     boolean compressionFlag =
+                         getBooleanAttribute(iTXt_node, "compressionFlag");
+-                    iTXt_compressionFlag.add(new Boolean(compressionFlag));
++                    iTXt_compressionFlag.add(Boolean.valueOf(compressionFlag));
+ 
+                     String compressionMethod =
+                         getAttribute(iTXt_node, "compressionMethod");
+-                    iTXt_compressionMethod.add(compressionMethod);
++                    iTXt_compressionMethod.add(Integer.valueOf(compressionMethod));
+ 
+                     String languageTag =
+                         getAttribute(iTXt_node, "languageTag");
+@@ -1952,13 +1950,10 @@
+                                 tEXt_text.add(value);
+                             }
+                         } else {
+-                            int flag = compression.equals("zip") ?
+-                                1 : 0;
+-
+                             // Use an iTXt node
+                             iTXt_keyword.add(keyword);
+-                            iTXt_compressionFlag.add(new Integer(flag));
+-                            iTXt_compressionMethod.add(new Integer(0));
++                            iTXt_compressionFlag.add(Boolean.valueOf(compression.equals("zip")));
++                            iTXt_compressionMethod.add(Integer.valueOf(0));
+                             iTXt_languageTag.add(language);
+                             iTXt_translatedKeyword.add(keyword); // fake it
+                             iTXt_text.add(value);
+@@ -1995,12 +1990,12 @@
+         gAMA_present = false;
+         hIST_present = false;
+         iCCP_present = false;
+-        iTXt_keyword = new ArrayList();
+-        iTXt_compressionFlag = new ArrayList();
+-        iTXt_compressionMethod = new ArrayList();
+-        iTXt_languageTag = new ArrayList();
+-        iTXt_translatedKeyword = new ArrayList();
+-        iTXt_text = new ArrayList();
++        iTXt_keyword = new ArrayList<String>();
++        iTXt_compressionFlag = new ArrayList<Boolean>();
++        iTXt_compressionMethod = new ArrayList<Integer>();
++        iTXt_languageTag = new ArrayList<String>();
++        iTXt_translatedKeyword = new ArrayList<String>();
++        iTXt_text = new ArrayList<String>();
+         pHYs_present = false;
+         sBIT_present = false;
+         sPLT_present = false;
+diff -urN openjdk.orig/jdk/test/javax/imageio/plugins/png/ITXtTest.java openjdk/jdk/test/javax/imageio/plugins/png/ITXtTest.java
+--- openjdk.orig/jdk/test/javax/imageio/plugins/png/ITXtTest.java	1969-12-31 19:00:00.000000000 -0500
++++ openjdk/jdk/test/javax/imageio/plugins/png/ITXtTest.java	2010-12-23 15:17:00.436446415 -0500
+@@ -0,0 +1,236 @@
++/*
++ * Copyright 2008 Sun Microsystems, Inc.  All Rights Reserved.
++ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
++ *
++ * This code is free software; you can redistribute it and/or modify it
++ * under the terms of the GNU General Public License version 2 only, as
++ * published by the Free Software Foundation.
++ *
++ * This code is distributed in the hope that it will be useful, but WITHOUT
++ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
++ * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
++ * version 2 for more details (a copy is included in the LICENSE file that
++ * accompanied this code).
++ *
++ * You should have received a copy of the GNU General Public License version
++ * 2 along with this work; if not, write to the Free Software Foundation,
++ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
++ *
++ * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
++ * CA 95054 USA or visit www.sun.com if you need additional information or
++ * have any questions.
++ */
++
++/**
++ * @test
++ * @bug     6541476
++ * @summary Test verifies that ImageIO PNG plugin correcly handles the
++ *          iTxt chunk (International textual data).
++ *
++ * @run     main ITXtTest
++ */
++
++
++import java.awt.Color;
++import java.awt.Graphics2D;
++import java.awt.image.BufferedImage;
++import java.io.File;
++
++import javax.imageio.ImageIO;
++import javax.imageio.ImageReader;
++import javax.imageio.IIOImage;
++import javax.imageio.ImageTypeSpecifier;
++import javax.imageio.ImageWriter;
++import javax.imageio.metadata.IIOMetadata;
++import javax.imageio.metadata.IIOMetadataNode;
++import javax.imageio.stream.ImageOutputStream;
++import javax.imageio.stream.ImageInputStream;
++
++import org.w3c.dom.Node;
++
++public class ITXtTest {
++    static public void main(String args[]) {
++        ITXtTest t_en = new ITXtTest();
++        t_en.description = "xml - en";
++        t_en.keyword = "XML:com.adobe.xmp";
++        t_en.isCompressed = false;
++        t_en.compression = 0;
++        t_en.language = "en";
++        t_en.trasKeyword = "XML:com.adobe.xmp";
++        t_en.text = "<xml>Something</xml>";
++
++        doTest(t_en);
++
++        // check compression case
++        t_en.isCompressed = true;
++        t_en.description = "xml - en - compressed";
++
++        doTest(t_en);
++
++        ITXtTest t_ru = new ITXtTest();
++        t_ru.description = "xml - ru";
++        t_ru.keyword = "XML:com.adobe.xmp";
++        t_ru.isCompressed = false;
++        t_ru.compression = 0;
++        t_ru.language = "ru";
++        t_ru.trasKeyword = "\u0410\u0410\u0410\u0410\u0410 XML";
++        t_ru.text = "<xml>\u042A\u042F\u042F\u042F\u042F\u042F\u042F</xml>";
++
++        doTest(t_ru);
++
++        t_ru.isCompressed = true;
++        t_ru.description = "xml - ru - compressed";
++
++        doTest(t_ru);
++    }
++
++
++    String description;
++
++    String keyword;
++    boolean isCompressed;
++    int compression;
++    String language;
++    String trasKeyword;
++    String text;
++
++
++    public IIOMetadataNode getNode() {
++        IIOMetadataNode iTXt = new IIOMetadataNode("iTXt");
++        IIOMetadataNode iTXtEntry = new IIOMetadataNode("iTXtEntry");
++        iTXtEntry.setAttribute("keyword", keyword);
++        iTXtEntry.setAttribute("compressionFlag",
++                               isCompressed ? "true" : "false");
++        iTXtEntry.setAttribute("compressionMethod",
++                               Integer.toString(compression));
++        iTXtEntry.setAttribute("languageTag", language);
++        iTXtEntry.setAttribute("translatedKeyword",
++                               trasKeyword);
++        iTXtEntry.setAttribute("text", text);
++        iTXt.appendChild(iTXtEntry);
++        return iTXt;
++    }
++
++    public static ITXtTest getFromNode(IIOMetadataNode n) {
++        ITXtTest t = new ITXtTest();
++
++        if (!"iTXt".equals(n.getNodeName())) {
++            throw new RuntimeException("Invalid node");
++        }
++        IIOMetadataNode e = (IIOMetadataNode)n.getFirstChild();
++        if (!"iTXtEntry".equals(e.getNodeName())) {
++            throw new RuntimeException("Invalid entry node");
++        }
++        t.keyword = e.getAttribute("keyword");
++        t.isCompressed =
++            (Integer.valueOf(e.getAttribute("compressionFlag")).intValue() == 1);
++        t.compression =
++            Integer.valueOf(e.getAttribute("compressionMethod")).intValue();
++        t.language = e.getAttribute("languageTag");
++        t.trasKeyword = e.getAttribute("translatedKeyword");
++        t.text = e.getAttribute("text");
++
++        return t;
++    }
++
++    @Override
++    public boolean equals(Object o) {
++        if (! (o instanceof ITXtTest)) {
++            return false;
++        }
++        ITXtTest t = (ITXtTest)o;
++        if (!keyword.equals(t.keyword)) { return false; }
++        if (isCompressed != t.isCompressed) { return false; }
++        if (compression != t.compression) { return false; }
++        if (!language.equals(t.language)) { return false; }
++        if (!trasKeyword.equals(t.trasKeyword)) { return false; }
++        if (!text.equals(t.text)) { return false; }
++
++        return true;
++    }
++
++
++
++    private static void doTest(ITXtTest src) {
++
++        System.out.println("Test: " + src.description);
++
++        File file = new File("test.png");
++
++        writeTo(file, src);
++        ITXtTest dst = readFrom(file);
++
++        if (dst == null || !dst.equals(src)) {
++            throw new RuntimeException("Test failed.");
++        }
++
++        System.out.println("Test passed.");
++    }
++
++    private static void writeTo(File f, ITXtTest t) {
++        BufferedImage src = createBufferedImage();
++        try {
++            ImageOutputStream imageOutputStream =
++                ImageIO.createImageOutputStream(f);
++
++            ImageTypeSpecifier imageTypeSpecifier =
++                new ImageTypeSpecifier(src);
++            ImageWriter imageWriter =
++                ImageIO.getImageWritersByFormatName("PNG").next();
++
++            imageWriter.setOutput(imageOutputStream);
++
++            IIOMetadata m =
++                imageWriter.getDefaultImageMetadata(imageTypeSpecifier, null);
++
++            String format = m.getNativeMetadataFormatName();
++            Node root = m.getAsTree(format);
++
++            IIOMetadataNode iTXt = t.getNode();
++            root.appendChild(iTXt);
++            m.setFromTree(format, root);
++
++            imageWriter.write(new IIOImage(src, null, m));
++            imageOutputStream.close();
++            System.out.println("Writing done.");
++        } catch (Throwable e) {
++            throw new RuntimeException("Writing test failed.", e);
++        }
++    }
++
++    private static ITXtTest readFrom(File f) {
++        try {
++            ImageInputStream iis = ImageIO.createImageInputStream(f);
++            ImageReader r = ImageIO.getImageReaders(iis).next();
++            r.setInput(iis);
++
++            IIOImage dst = r.readAll(0, null);
++
++            // look for iTXt node
++            IIOMetadata m = dst.getMetadata();
++            Node root = m.getAsTree(m.getNativeMetadataFormatName());
++            Node n = root.getFirstChild();
++            while (n != null && !"iTXt".equals(n.getNodeName())) {
++                n = n.getNextSibling();
++            }
++            if (n == null) {
++                throw new RuntimeException("No iTXt node!");
++            }
++            ITXtTest t = ITXtTest.getFromNode((IIOMetadataNode)n);
++            return t;
++        } catch (Throwable e) {
++            throw new RuntimeException("Reading test failed.", e);
++        }
++    }
++
++    private static BufferedImage createBufferedImage() {
++        BufferedImage image = new BufferedImage(128, 128,
++                      BufferedImage.TYPE_4BYTE_ABGR_PRE);
++        Graphics2D graph = image.createGraphics();
++        graph.setPaintMode();
++        graph.setColor(Color.orange);
++        graph.fillRect(32, 32, 64, 64);
++        graph.dispose();
++        return image;
++    }
++}
diff -r 7219bd74962a patches/openjdk/6687968-pngimagereader_mem_leak.patch
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/patches/openjdk/6687968-pngimagereader_mem_leak.patch	Thu Dec 23 15:20:57 2010 -0500
@@ -0,0 +1,126 @@
+diff -urN openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageReader.java openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageReader.java
+--- openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageReader.java	2010-06-21 17:15:11.000000000 -0400
++++ openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageReader.java	2010-12-23 15:12:24.471446202 -0500
+@@ -618,10 +618,15 @@
+     private static String inflate(byte[] b) throws IOException {
+         InputStream bais = new ByteArrayInputStream(b);
+         InputStream iis = new InflaterInputStream(bais);
++
+         StringBuilder sb = new StringBuilder(80);
+         int c;
+-        while ((c = iis.read()) != -1) {
+-            sb.append((char)c);
++        try {
++            while ((c = iis.read()) != -1) {
++                sb.append((char)c);
++            }
++        } finally {
++            iis.close();
+         }
+         return sb.toString();
+     }
+@@ -1246,13 +1251,26 @@
+             destinationBands = param.getDestinationBands();
+             destinationOffset = param.getDestinationOffset();
+         }
+-
++        Inflater inf = null;
+         try {
+             stream.seek(imageStartPosition);
+ 
+             Enumeration e = new PNGImageDataEnumeration(stream);
+             InputStream is = new SequenceInputStream(e);
+-            is = new InflaterInputStream(is, new Inflater());
++
++           /* InflaterInputStream uses an Inflater instance which consumes
++            * native (non-GC visible) resources. This is normally implicitly
++            * freed when the stream is closed. However since the
++            * InflaterInputStream wraps a client-supplied input stream,
++            * we cannot close it.
++            * But the app may depend on GC finalization to close the stream.
++            * Therefore to ensure timely freeing of native resources we
++            * explicitly create the Inflater instance and free its resources
++            * when we are done with the InflaterInputStream by calling
++            * inf.end();
++            */
++            inf = new Inflater();
++            is = new InflaterInputStream(is, inf);
+             is = new BufferedInputStream(is);
+             this.pixelStream = new DataInputStream(is);
+ 
+@@ -1285,6 +1303,10 @@
+             }
+         } catch (IOException e) {
+             throw new IIOException("Error reading PNG image data", e);
++        } finally {
++            if (inf != null) {
++                inf.end();
++            }
+         }
+     }
+ 
+diff -urN openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageWriter.java openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageWriter.java
+--- openjdk.orig/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageWriter.java	2010-06-21 17:15:11.000000000 -0400
++++ openjdk/jdk/src/share/classes/com/sun/imageio/plugins/png/PNGImageWriter.java	2010-12-23 15:12:24.472446225 -0500
+@@ -244,13 +244,17 @@
+     }
+ 
+     public void finish() throws IOException {
+-        if (!def.finished()) {
+-            def.finish();
+-            while (!def.finished()) {
+-                deflate();
++        try {
++            if (!def.finished()) {
++                def.finish();
++                while (!def.finished()) {
++                    deflate();
++                }
+             }
++            finishChunk();
++        } finally {
++            def.end();
+         }
+-        finishChunk();
+     }
+ 
+     protected void finalize() throws Throwable {
+@@ -928,23 +932,24 @@
+     // Use sourceXOffset, etc.
+     private void write_IDAT(RenderedImage image) throws IOException {
+         IDATOutputStream ios = new IDATOutputStream(stream, 32768);
+-
+-        if (metadata.IHDR_interlaceMethod == 1) {
+-            for (int i = 0; i < 7; i++) {
+-                encodePass(ios, image,
+-                           PNGImageReader.adam7XOffset[i],
+-                           PNGImageReader.adam7YOffset[i],
+-                           PNGImageReader.adam7XSubsampling[i],
+-                           PNGImageReader.adam7YSubsampling[i]);
+-                if (abortRequested()) {
+-                    break;
++        try {
++            if (metadata.IHDR_interlaceMethod == 1) {
++                for (int i = 0; i < 7; i++) {
++                    encodePass(ios, image,
++                               PNGImageReader.adam7XOffset[i],
++                               PNGImageReader.adam7YOffset[i],
++                               PNGImageReader.adam7XSubsampling[i],
++                               PNGImageReader.adam7YSubsampling[i]);
++                    if (abortRequested()) {
++                        break;
++                    }
+                 }
++            } else {
++                encodePass(ios, image, 0, 0, 1, 1);
+             }
+-        } else {
+-            encodePass(ios, image, 0, 0, 1, 1);
++        } finally {
++            ios.finish();
+         }
+-
+-        ios.finish();
+     }
+ 
+     private void writeIEND() throws IOException {


More information about the distro-pkg-dev mailing list