5b1765f25d
Reset stream position after abort; change IAEs to NPEs. Reviewed-by: prr, serb
225 lines
7.3 KiB
Java
225 lines
7.3 KiB
Java
/*
|
|
* Copyright (c) 2015, Oracle and/or its affiliates. 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 Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
import java.awt.Color;
|
|
import java.awt.Graphics2D;
|
|
import java.awt.image.BufferedImage;
|
|
import java.io.File;
|
|
import java.io.FileNotFoundException;
|
|
import java.io.FileOutputStream;
|
|
import java.io.IOException;
|
|
import java.util.Iterator;
|
|
|
|
import javax.imageio.ImageIO;
|
|
import javax.imageio.ImageWriter;
|
|
import javax.imageio.event.IIOWriteProgressListener;
|
|
import javax.imageio.spi.IIORegistry;
|
|
import javax.imageio.spi.ImageWriterSpi;
|
|
import javax.imageio.stream.ImageOutputStream;
|
|
|
|
import static java.awt.image.BufferedImage.TYPE_BYTE_BINARY;
|
|
|
|
/**
|
|
* @test
|
|
* @bug 4952954
|
|
* @summary abortFlag must be cleared for every ImageWriter.write operation
|
|
* @author Sergey Bylokhov
|
|
*/
|
|
public final class WriteAfterAbort implements IIOWriteProgressListener {
|
|
|
|
private volatile boolean abortFlag = true;
|
|
private volatile boolean isAbortCalled;
|
|
private volatile boolean isCompleteCalled;
|
|
private volatile boolean isProgressCalled;
|
|
private volatile boolean isStartedCalled;
|
|
private static final int WIDTH = 100;
|
|
private static final int HEIGHT = 100;
|
|
|
|
private void test(final ImageWriter writer) throws IOException {
|
|
// Image initialization
|
|
final BufferedImage imageWrite = new BufferedImage(WIDTH, HEIGHT,
|
|
TYPE_BYTE_BINARY);
|
|
final Graphics2D g = imageWrite.createGraphics();
|
|
g.setColor(Color.WHITE);
|
|
g.fillRect(0, 0, WIDTH, HEIGHT);
|
|
g.dispose();
|
|
|
|
// File initialization
|
|
final File file = File.createTempFile("temp", ".img");
|
|
file.deleteOnExit();
|
|
final FileOutputStream fos = new SkipWriteOnAbortOutputStream(file);
|
|
final ImageOutputStream ios = ImageIO.createImageOutputStream(fos);
|
|
writer.setOutput(ios);
|
|
writer.addIIOWriteProgressListener(this);
|
|
|
|
// This write will be aborted, and file will not be touched
|
|
writer.write(imageWrite);
|
|
if (!isStartedCalled) {
|
|
throw new RuntimeException("Started should be called");
|
|
}
|
|
if (!isProgressCalled) {
|
|
throw new RuntimeException("Progress should be called");
|
|
}
|
|
if (!isAbortCalled) {
|
|
throw new RuntimeException("Abort should be called");
|
|
}
|
|
if (isCompleteCalled) {
|
|
throw new RuntimeException("Complete should not be called");
|
|
}
|
|
// Flush aborted data
|
|
ios.flush();
|
|
|
|
// This write should be completed successfully and the file should
|
|
// contain correct image data.
|
|
abortFlag = false;
|
|
isAbortCalled = false;
|
|
isCompleteCalled = false;
|
|
isProgressCalled = false;
|
|
isStartedCalled = false;
|
|
writer.write(imageWrite);
|
|
|
|
if (!isStartedCalled) {
|
|
throw new RuntimeException("Started should be called");
|
|
}
|
|
if (!isProgressCalled) {
|
|
throw new RuntimeException("Progress should be called");
|
|
}
|
|
if (isAbortCalled) {
|
|
throw new RuntimeException("Abort should not be called");
|
|
}
|
|
if (!isCompleteCalled) {
|
|
throw new RuntimeException("Complete should be called");
|
|
}
|
|
writer.dispose();
|
|
ios.close();
|
|
|
|
// Validates content of the file.
|
|
final BufferedImage imageRead = ImageIO.read(file);
|
|
for (int x = 0; x < WIDTH; ++x) {
|
|
for (int y = 0; y < HEIGHT; ++y) {
|
|
if (imageRead.getRGB(x, y) != imageWrite.getRGB(x, y)) {
|
|
throw new RuntimeException("Test failed.");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public static void main(final String[] args) throws IOException {
|
|
final IIORegistry registry = IIORegistry.getDefaultInstance();
|
|
final Iterator<ImageWriterSpi> iter = registry.getServiceProviders(
|
|
ImageWriterSpi.class, provider -> true, true);
|
|
|
|
// Validates all supported ImageWriters
|
|
int numFailures = 0;
|
|
while (iter.hasNext()) {
|
|
final WriteAfterAbort writeAfterAbort = new WriteAfterAbort();
|
|
final ImageWriter writer = iter.next().createWriterInstance();
|
|
System.out.println("ImageWriter = " + writer);
|
|
try {
|
|
writeAfterAbort.test(writer);
|
|
} catch (Exception e) {
|
|
System.err.println("Test failed for \""
|
|
+ writer.getOriginatingProvider().getFormatNames()[0]
|
|
+ "\" format.");
|
|
numFailures++;
|
|
}
|
|
}
|
|
if (numFailures == 0) {
|
|
System.out.println("Test passed.");
|
|
} else {
|
|
throw new RuntimeException("Test failed.");
|
|
}
|
|
}
|
|
|
|
// Callbacks
|
|
|
|
@Override
|
|
public void imageComplete(ImageWriter source) {
|
|
isCompleteCalled = true;
|
|
}
|
|
|
|
@Override
|
|
public void imageProgress(ImageWriter source, float percentageDone) {
|
|
isProgressCalled = true;
|
|
if (percentageDone > 50 && abortFlag) {
|
|
source.abort();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void imageStarted(ImageWriter source, int imageIndex) {
|
|
isStartedCalled = true;
|
|
}
|
|
|
|
@Override
|
|
public void writeAborted(final ImageWriter source) {
|
|
isAbortCalled = true;
|
|
}
|
|
|
|
@Override
|
|
public void thumbnailComplete(ImageWriter source) {
|
|
}
|
|
|
|
@Override
|
|
public void thumbnailProgress(ImageWriter source, float percentageDone) {
|
|
}
|
|
|
|
@Override
|
|
public void thumbnailStarted(ImageWriter source, int imageIndex,
|
|
int thumbnailIndex) {
|
|
}
|
|
|
|
/**
|
|
* We need to skip writes on abort, because content of the file after abort
|
|
* is undefined.
|
|
*/
|
|
private class SkipWriteOnAbortOutputStream extends FileOutputStream {
|
|
|
|
SkipWriteOnAbortOutputStream(File file) throws FileNotFoundException {
|
|
super(file);
|
|
}
|
|
|
|
@Override
|
|
public void write(int b) throws IOException {
|
|
if (!abortFlag) {
|
|
super.write(b);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void write(byte[] b) throws IOException {
|
|
if (!abortFlag) {
|
|
super.write(b);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void write(byte[] b, int off, int len) throws IOException {
|
|
if (!abortFlag) {
|
|
super.write(b, off, len);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|