/* * Copyright (C)2011 D. R. Commander. All Rights Reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * - Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * - Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * - Neither the name of the libjpeg-turbo Project nor the names of its * contributors may be used to endorse or promote products derived from this * software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS", * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ package org.libjpegturbo.turbojpeg; import java.awt.image.*; public class TJDecompressor { public TJDecompressor() throws Exception { init(); } public TJDecompressor(byte [] buf) throws Exception { setJPEGBuffer(buf, buf.length); } public TJDecompressor(byte [] buf, int bufSize) throws Exception { setJPEGBuffer(buf, bufSize); } public void setJPEGBuffer(byte [] buf, int bufSize) throws Exception { if(handle == 0) init(); if(buf == null || bufSize < 1) throw new Exception("Invalid argument in setJPEGBuffer()"); jpegBuf = buf; jpegBufSize = bufSize; decompressHeader(jpegBuf, jpegBufSize); } public int getWidth() throws Exception { if(jpegWidth < 1) throw new Exception("JPEG buffer not initialized"); return jpegWidth; } public int getHeight() throws Exception { if(jpegHeight < 1) throw new Exception("JPEG buffer not initialized"); return jpegHeight; } public int getSubsamp() throws Exception { if(jpegSubsamp < 0) throw new Exception("JPEG buffer not initialized"); if(jpegSubsamp >= TJ.NUMSAMPOPT) throw new Exception("JPEG header information is invalid"); return jpegSubsamp; } public int getScaledWidth(int desiredWidth, int desiredHeight) throws Exception { if(jpegWidth < 1 || jpegHeight < 1) throw new Exception("JPEG buffer not initialized"); if(desiredWidth < 0 || desiredHeight < 0) throw new Exception("Invalid argument in getScaledWidth()"); TJ.ScalingFactor sf [] = TJ.getScalingFactors(); if(desiredWidth == 0) desiredWidth = jpegWidth; if(desiredHeight == 0) desiredHeight = jpegHeight; int scaledWidth = jpegWidth, scaledHeight = jpegHeight; for(int i=0; i desiredWidth || scaledHeight > desiredHeight) throw new Exception("Could not scale down to desired image dimensions"); return scaledWidth; } public int getScaledHeight(int desiredWidth, int desiredHeight) throws Exception { if(jpegWidth < 1 || jpegHeight < 1) throw new Exception("JPEG buffer not initialized"); if(desiredWidth < 0 || desiredHeight < 0) throw new Exception("Invalid argument in getScaledHeight()"); TJ.ScalingFactor sf [] = TJ.getScalingFactors(); if(desiredWidth == 0) desiredWidth = jpegWidth; if(desiredHeight == 0) desiredHeight = jpegHeight; int scaledWidth = jpegWidth, scaledHeight = jpegHeight; for(int i=0; i desiredWidth || scaledHeight > desiredHeight) throw new Exception("Could not scale down to desired image dimensions"); return scaledHeight; } public void decompress(byte [] dstBuf, int desiredWidth, int pitch, int desiredHeight, int pixelFormat, int flags) throws Exception { if(jpegBuf == null) throw new Exception("JPEG buffer not initialized"); if(dstBuf == null || desiredWidth < 0 || pitch < 0 || desiredHeight < 0 || pixelFormat < 0 || pixelFormat >= TJ.NUMPFOPT || flags < 0) throw new Exception("Invalid argument in decompress()"); decompress(jpegBuf, jpegBufSize, dstBuf, desiredWidth, pitch, desiredHeight, pixelFormat, flags); } public byte [] decompress(int desiredWidth, int pitch, int desiredHeight, int pixelFormat, int flags) throws Exception { if(desiredWidth < 0 || pitch < 0 || desiredHeight < 0 || pixelFormat < 0 || pixelFormat >= TJ.NUMPFOPT || flags < 0) throw new Exception("Invalid argument in decompress()"); int pixelSize = TJ.getPixelSize(pixelFormat); int scaledWidth = getScaledWidth(desiredWidth, desiredHeight); int scaledHeight = getScaledHeight(desiredWidth, desiredHeight); if(pitch == 0) pitch = scaledWidth * pixelSize; byte [] buf = new byte[pitch * scaledHeight]; decompress(buf, desiredWidth, pitch, desiredHeight, pixelFormat, flags); return buf; } public void decompressToYUV(byte [] dstBuf, int flags) throws Exception { if(jpegBuf == null) throw new Exception("JPEG buffer not initialized"); if(dstBuf == null || flags < 0) throw new Exception("Invalid argument in decompressToYUV()"); decompressToYUV(jpegBuf, jpegBufSize, dstBuf, flags); } public byte [] decompressToYUV(int flags) throws Exception { if(flags < 0) throw new Exception("Invalid argument in decompressToYUV()"); if(jpegWidth < 1 || jpegHeight < 1 || jpegSubsamp < 0) throw new Exception("JPEG buffer not initialized"); if(jpegSubsamp >= TJ.NUMSAMPOPT) throw new Exception("JPEG header information is invalid"); byte [] buf = new byte[TJ.bufSizeYUV(jpegWidth, jpegHeight, jpegSubsamp)]; decompressToYUV(buf, flags); return buf; } public void decompress(BufferedImage dstImage, int flags) throws Exception { if(dstImage == null || flags < 0) throw new Exception("Invalid argument in decompress()"); int desiredWidth = dstImage.getWidth(); int desiredHeight = dstImage.getHeight(); int scaledWidth = getScaledWidth(desiredWidth, desiredHeight); int scaledHeight = getScaledHeight(desiredWidth, desiredHeight); if(scaledWidth != desiredWidth || scaledHeight != desiredHeight) throw new Exception("BufferedImage dimensions do not match a scaled image size that TurboJPEG is capable of generating."); int pixelFormat; boolean intPixels=false; switch(dstImage.getType()) { case BufferedImage.TYPE_3BYTE_BGR: pixelFormat=TJ.PF_BGR; break; case BufferedImage.TYPE_BYTE_GRAY: pixelFormat=TJ.PF_GRAY; break; case BufferedImage.TYPE_INT_BGR: pixelFormat=TJ.PF_RGBX; intPixels=true; break; case BufferedImage.TYPE_INT_RGB: pixelFormat=TJ.PF_BGRX; intPixels=true; break; default: throw new Exception("Unsupported BufferedImage format"); } WritableRaster wr = dstImage.getRaster(); if(intPixels) { SinglePixelPackedSampleModel sm = (SinglePixelPackedSampleModel)dstImage.getSampleModel(); int pitch = sm.getScanlineStride(); DataBufferInt db = (DataBufferInt)wr.getDataBuffer(); int [] buf = db.getData(); if(jpegBuf == null) throw new Exception("JPEG buffer not initialized"); decompress(jpegBuf, jpegBufSize, buf, scaledWidth, pitch, scaledHeight, pixelFormat, flags); } else { ComponentSampleModel sm = (ComponentSampleModel)dstImage.getSampleModel(); int pixelSize = sm.getPixelStride(); if(pixelSize != TJ.getPixelSize(pixelFormat)) throw new Exception("Inconsistency between pixel format and pixel size in BufferedImage"); int pitch = sm.getScanlineStride(); DataBufferByte db = (DataBufferByte)wr.getDataBuffer(); byte [] buf = db.getData(); decompress(buf, scaledWidth, pitch, scaledHeight, pixelFormat, flags); } } public BufferedImage decompress(int desiredWidth, int desiredHeight, int bufferedImageType, int flags) throws Exception { if(desiredWidth < 0 || desiredHeight < 0 || flags < 0) throw new Exception("Invalid argument in decompress()"); int scaledWidth = getScaledWidth(desiredWidth, desiredHeight); int scaledHeight = getScaledHeight(desiredWidth, desiredHeight); BufferedImage img = new BufferedImage(scaledWidth, scaledHeight, bufferedImageType); decompress(img, flags); return img; } public void close() throws Exception { destroy(); } protected void finalize() throws Throwable { try { close(); } catch(Exception e) { } finally { super.finalize(); } }; private native void init() throws Exception; private native void destroy() throws Exception; private native void decompressHeader(byte [] srcBuf, int size) throws Exception; private native void decompress(byte [] srcBuf, int size, byte [] dstBuf, int desiredWidth, int pitch, int desiredHeight, int pixelFormat, int flags) throws Exception; private native void decompress(byte [] srcBuf, int size, int [] dstBuf, int desiredWidth, int pitch, int desiredHeight, int pixelFormat, int flags) throws Exception; private native void decompressToYUV(byte [] srcBuf, int size, byte [] dstBuf, int flags) throws Exception; static { System.loadLibrary("turbojpeg"); } private long handle = 0; private byte [] jpegBuf = null; private int jpegBufSize = 0; private int jpegWidth = 0; private int jpegHeight = 0; private int jpegSubsamp = -1; };