1 /*
2  * Copyright (C)2011-2015 D. R. Commander.  All Rights Reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are met:
6  *
7  * - Redistributions of source code must retain the above copyright notice,
8  *   this list of conditions and the following disclaimer.
9  * - Redistributions in binary form must reproduce the above copyright notice,
10  *   this list of conditions and the following disclaimer in the documentation
11  *   and/or other materials provided with the distribution.
12  * - Neither the name of the libjpeg-turbo Project nor the names of its
13  *   contributors may be used to endorse or promote products derived from this
14  *   software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS",
17  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
20  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26  * POSSIBILITY OF SUCH DAMAGE.
27  */
28 
29 package org.libjpegturbo.turbojpeg;
30 
31 import java.awt.image.*;
32 import java.nio.*;
33 
34 /**
35  * TurboJPEG compressor
36  */
37 public class TJCompressor {
38 
39   private static final String NO_ASSOC_ERROR =
40     "No source image is associated with this instance";
41 
42   /**
43    * Create a TurboJPEG compressor instance.
44    */
TJCompressor()45   public TJCompressor() throws Exception {
46     init();
47   }
48 
49   /**
50    * Create a TurboJPEG compressor instance and associate the uncompressed
51    * source image stored in <code>srcImage</code> with the newly created
52    * instance.
53    *
54    * @param srcImage see {@link #setSourceImage} for description
55    *
56    * @param x see {@link #setSourceImage} for description
57    *
58    * @param y see {@link #setSourceImage} for description
59    *
60    * @param width see {@link #setSourceImage} for description
61    *
62    * @param pitch see {@link #setSourceImage} for description
63    *
64    * @param height see {@link #setSourceImage} for description
65    *
66    * @param pixelFormat pixel format of the source image (one of
67    * {@link TJ#PF_RGB TJ.PF_*})
68    */
TJCompressor(byte[] srcImage, int x, int y, int width, int pitch, int height, int pixelFormat)69   public TJCompressor(byte[] srcImage, int x, int y, int width, int pitch,
70                       int height, int pixelFormat) throws Exception {
71     setSourceImage(srcImage, x, y, width, pitch, height, pixelFormat);
72   }
73 
74   /**
75    * @deprecated Use
76    * {@link #TJCompressor(byte[], int, int, int, int, int, int)} instead.
77    */
78   @Deprecated
TJCompressor(byte[] srcImage, int width, int pitch, int height, int pixelFormat)79   public TJCompressor(byte[] srcImage, int width, int pitch, int height,
80                       int pixelFormat) throws Exception {
81     setSourceImage(srcImage, width, pitch, height, pixelFormat);
82   }
83 
84   /**
85    * Create a TurboJPEG compressor instance and associate the uncompressed
86    * source image stored in <code>srcImage</code> with the newly created
87    * instance.
88    *
89    * @param srcImage see
90    * {@link #setSourceImage(BufferedImage, int, int, int, int)} for description
91    *
92    * @param x see
93    * {@link #setSourceImage(BufferedImage, int, int, int, int)} for description
94    *
95    * @param y see
96    * {@link #setSourceImage(BufferedImage, int, int, int, int)} for description
97    *
98    * @param width see
99    * {@link #setSourceImage(BufferedImage, int, int, int, int)} for description
100    *
101    * @param height see
102    * {@link #setSourceImage(BufferedImage, int, int, int, int)} for description
103    */
TJCompressor(BufferedImage srcImage, int x, int y, int width, int height)104   public TJCompressor(BufferedImage srcImage, int x, int y, int width,
105                       int height) throws Exception {
106     setSourceImage(srcImage, x, y, width, height);
107   }
108 
109   /**
110    * Associate an uncompressed RGB, grayscale, or CMYK source image with this
111    * compressor instance.
112    *
113    * @param srcImage image buffer containing RGB, grayscale, or CMYK pixels to
114    * be compressed or encoded.  This buffer is not modified.
115    *
116    * @param x x offset (in pixels) of the region in the source image from which
117    * the JPEG or YUV image should be compressed/encoded
118    *
119    * @param y y offset (in pixels) of the region in the source image from which
120    * the JPEG or YUV image should be compressed/encoded
121    *
122    * @param width width (in pixels) of the region in the source image from
123    * which the JPEG or YUV image should be compressed/encoded
124    *
125    * @param pitch bytes per line of the source image.  Normally, this should be
126    * <code>width * TJ.pixelSize(pixelFormat)</code> if the source image is
127    * unpadded, but you can use this parameter to, for instance, specify that
128    * the scanlines in the source image are padded to a 4-byte boundary or to
129    * compress/encode a JPEG or YUV image from a region of a larger source
130    * image.  You can also be clever and use this parameter to skip lines, etc.
131    * Setting this parameter to 0 is the equivalent of setting it to
132    * <code>width * TJ.pixelSize(pixelFormat)</code>.
133    *
134    * @param height height (in pixels) of the region in the source image from
135    * which the JPEG or YUV image should be compressed/encoded
136    *
137    * @param pixelFormat pixel format of the source image (one of
138    * {@link TJ#PF_RGB TJ.PF_*})
139    */
setSourceImage(byte[] srcImage, int x, int y, int width, int pitch, int height, int pixelFormat)140   public void setSourceImage(byte[] srcImage, int x, int y, int width,
141                              int pitch, int height, int pixelFormat)
142                              throws Exception {
143     if (handle == 0) init();
144     if (srcImage == null || x < 0 || y < 0 || width < 1 || height < 1 ||
145         pitch < 0 || pixelFormat < 0 || pixelFormat >= TJ.NUMPF)
146       throw new Exception("Invalid argument in setSourceImage()");
147     srcBuf = srcImage;
148     srcWidth = width;
149     if (pitch == 0)
150       srcPitch = width * TJ.getPixelSize(pixelFormat);
151     else
152       srcPitch = pitch;
153     srcHeight = height;
154     srcPixelFormat = pixelFormat;
155     srcX = x;
156     srcY = y;
157     srcBufInt = null;
158     srcYUVImage = null;
159   }
160 
161   /**
162    * @deprecated Use
163    * {@link #setSourceImage(byte[], int, int, int, int, int, int)} instead.
164    */
165   @Deprecated
setSourceImage(byte[] srcImage, int width, int pitch, int height, int pixelFormat)166   public void setSourceImage(byte[] srcImage, int width, int pitch,
167                              int height, int pixelFormat) throws Exception {
168     setSourceImage(srcImage, 0, 0, width, pitch, height, pixelFormat);
169     srcX = srcY = -1;
170   }
171 
172   /**
173    * Associate an uncompressed RGB or grayscale source image with this
174    * compressor instance.
175    *
176    * @param srcImage a <code>BufferedImage</code> instance containing RGB or
177    * grayscale pixels to be compressed or encoded.  This image is not modified.
178    *
179    * @param x x offset (in pixels) of the region in the source image from which
180    * the JPEG or YUV image should be compressed/encoded
181    *
182    * @param y y offset (in pixels) of the region in the source image from which
183    * the JPEG or YUV image should be compressed/encoded
184    *
185    * @param width width (in pixels) of the region in the source image from
186    * which the JPEG or YUV image should be compressed/encoded (0 = use the
187    * width of the source image)
188    *
189    * @param height height (in pixels) of the region in the source image from
190    * which the JPEG or YUV image should be compressed/encoded (0 = use the
191    * height of the source image)
192    */
setSourceImage(BufferedImage srcImage, int x, int y, int width, int height)193   public void setSourceImage(BufferedImage srcImage, int x, int y, int width,
194                              int height) throws Exception {
195     if (handle == 0) init();
196     if (srcImage == null || x < 0 || y < 0 || width < 0 || height < 0)
197       throw new Exception("Invalid argument in setSourceImage()");
198     srcX = x;
199     srcY = y;
200     srcWidth = (width == 0) ? srcImage.getWidth(): width;
201     srcHeight = (height == 0) ? srcImage.getHeight() : height;
202     if (x + width > srcImage.getWidth() || y + height > srcImage.getHeight())
203       throw new Exception("Compression region exceeds the bounds of the source image");
204 
205     int pixelFormat;
206     boolean intPixels = false;
207     if (byteOrder == null)
208       byteOrder = ByteOrder.nativeOrder();
209     switch(srcImage.getType()) {
210       case BufferedImage.TYPE_3BYTE_BGR:
211         pixelFormat = TJ.PF_BGR;  break;
212       case BufferedImage.TYPE_4BYTE_ABGR:
213       case BufferedImage.TYPE_4BYTE_ABGR_PRE:
214         pixelFormat = TJ.PF_XBGR;  break;
215       case BufferedImage.TYPE_BYTE_GRAY:
216         pixelFormat = TJ.PF_GRAY;  break;
217       case BufferedImage.TYPE_INT_BGR:
218         if (byteOrder == ByteOrder.BIG_ENDIAN)
219           pixelFormat = TJ.PF_XBGR;
220         else
221           pixelFormat = TJ.PF_RGBX;
222         intPixels = true;  break;
223       case BufferedImage.TYPE_INT_RGB:
224       case BufferedImage.TYPE_INT_ARGB:
225       case BufferedImage.TYPE_INT_ARGB_PRE:
226         if (byteOrder == ByteOrder.BIG_ENDIAN)
227           pixelFormat = TJ.PF_XRGB;
228         else
229           pixelFormat = TJ.PF_BGRX;
230         intPixels = true;  break;
231       default:
232         throw new Exception("Unsupported BufferedImage format");
233     }
234     srcPixelFormat = pixelFormat;
235 
236     WritableRaster wr = srcImage.getRaster();
237     if (intPixels) {
238       SinglePixelPackedSampleModel sm =
239         (SinglePixelPackedSampleModel)srcImage.getSampleModel();
240       srcStride = sm.getScanlineStride();
241       DataBufferInt db = (DataBufferInt)wr.getDataBuffer();
242       srcBufInt = db.getData();
243       srcBuf = null;
244     } else {
245       ComponentSampleModel sm =
246         (ComponentSampleModel)srcImage.getSampleModel();
247       int pixelSize = sm.getPixelStride();
248       if (pixelSize != TJ.getPixelSize(pixelFormat))
249         throw new Exception("Inconsistency between pixel format and pixel size in BufferedImage");
250       srcPitch = sm.getScanlineStride();
251       DataBufferByte db = (DataBufferByte)wr.getDataBuffer();
252       srcBuf = db.getData();
253       srcBufInt = null;
254     }
255     srcYUVImage = null;
256   }
257 
258   /**
259    * Associate an uncompressed YUV planar source image with this compressor
260    * instance.
261    *
262    * @param srcImage YUV planar image to be compressed.  This image is not
263    * modified.
264    */
setSourceImage(YUVImage srcImage)265   public void setSourceImage(YUVImage srcImage) throws Exception {
266     if (handle == 0) init();
267     if (srcImage == null)
268       throw new Exception("Invalid argument in setSourceImage()");
269     srcYUVImage = srcImage;
270     srcBuf = null;
271     srcBufInt = null;
272   }
273 
274   /**
275    * Set the level of chrominance subsampling for subsequent compress/encode
276    * operations.  When pixels are converted from RGB to YCbCr (see
277    * {@link TJ#CS_YCbCr}) or from CMYK to YCCK (see {@link TJ#CS_YCCK}) as part
278    * of the JPEG compression process, some of the Cb and Cr (chrominance)
279    * components can be discarded or averaged together to produce a smaller
280    * image with little perceptible loss of image clarity (the human eye is more
281    * sensitive to small changes in brightness than to small changes in color.)
282    * This is called "chrominance subsampling".
283    * <p>
284    * NOTE: This method has no effect when compressing a JPEG image from a YUV
285    * planar source.  In that case, the level of chrominance subsampling in
286    * the JPEG image is determined by the source.  Further, this method has no
287    * effect when encoding to a pre-allocated {@link YUVImage} instance.  In
288    * that case, the level of chrominance subsampling is determined by the
289    * destination.
290    *
291    * @param newSubsamp the level of chrominance subsampling to use in
292    * subsequent compress/encode oeprations (one of
293    * {@link TJ#SAMP_444 TJ.SAMP_*})
294    */
setSubsamp(int newSubsamp)295   public void setSubsamp(int newSubsamp) throws Exception {
296     if (newSubsamp < 0 || newSubsamp >= TJ.NUMSAMP)
297       throw new Exception("Invalid argument in setSubsamp()");
298     subsamp = newSubsamp;
299   }
300 
301   /**
302    * Set the JPEG image quality level for subsequent compress operations.
303    *
304    * @param quality the new JPEG image quality level (1 to 100, 1 = worst,
305    * 100 = best)
306    */
setJPEGQuality(int quality)307   public void setJPEGQuality(int quality) throws Exception {
308     if (quality < 1 || quality > 100)
309       throw new Exception("Invalid argument in setJPEGQuality()");
310     jpegQuality = quality;
311   }
312 
313   /**
314    * Compress the uncompressed source image associated with this compressor
315    * instance and output a JPEG image to the given destination buffer.
316    *
317    * @param dstBuf buffer that will receive the JPEG image.  Use
318    * {@link TJ#bufSize} to determine the maximum size for this buffer based on
319    * the source image's width and height and the desired level of chrominance
320    * subsampling.
321    *
322    * @param flags the bitwise OR of one or more of
323    * {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
324    */
compress(byte[] dstBuf, int flags)325   public void compress(byte[] dstBuf, int flags) throws Exception {
326     if (dstBuf == null || flags < 0)
327       throw new Exception("Invalid argument in compress()");
328     if (srcBuf == null && srcBufInt == null && srcYUVImage == null)
329       throw new Exception(NO_ASSOC_ERROR);
330     if (jpegQuality < 0)
331       throw new Exception("JPEG Quality not set");
332     if (subsamp < 0 && srcYUVImage == null)
333       throw new Exception("Subsampling level not set");
334 
335     if (srcYUVImage != null)
336       compressedSize = compressFromYUV(srcYUVImage.getPlanes(),
337                                        srcYUVImage.getOffsets(),
338                                        srcYUVImage.getWidth(),
339                                        srcYUVImage.getStrides(),
340                                        srcYUVImage.getHeight(),
341                                        srcYUVImage.getSubsamp(),
342                                        dstBuf, jpegQuality, flags);
343     else if (srcBuf != null) {
344       if (srcX >= 0 && srcY >= 0)
345         compressedSize = compress(srcBuf, srcX, srcY, srcWidth, srcPitch,
346                                   srcHeight, srcPixelFormat, dstBuf, subsamp,
347                                   jpegQuality, flags);
348       else
349         compressedSize = compress(srcBuf, srcWidth, srcPitch, srcHeight,
350                                   srcPixelFormat, dstBuf, subsamp, jpegQuality,
351                                   flags);
352     } else if (srcBufInt != null) {
353       if (srcX >= 0 && srcY >= 0)
354         compressedSize = compress(srcBufInt, srcX, srcY, srcWidth, srcStride,
355                                   srcHeight, srcPixelFormat, dstBuf, subsamp,
356                                   jpegQuality, flags);
357       else
358         compressedSize = compress(srcBufInt, srcWidth, srcStride, srcHeight,
359                                   srcPixelFormat, dstBuf, subsamp, jpegQuality,
360                                   flags);
361     }
362   }
363 
364   /**
365    * Compress the uncompressed source image associated with this compressor
366    * instance and return a buffer containing a JPEG image.
367    *
368    * @param flags the bitwise OR of one or more of
369    * {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
370    *
371    * @return a buffer containing a JPEG image.  The length of this buffer will
372    * not be equal to the size of the JPEG image.  Use {@link
373    * #getCompressedSize} to obtain the size of the JPEG image.
374    */
compress(int flags)375   public byte[] compress(int flags) throws Exception {
376     if (srcWidth < 1 || srcHeight < 1)
377       throw new Exception(NO_ASSOC_ERROR);
378     byte[] buf = new byte[TJ.bufSize(srcWidth, srcHeight, subsamp)];
379     compress(buf, flags);
380     return buf;
381   }
382 
383   /**
384    * @deprecated Use
385    * {@link #setSourceImage(BufferedImage, int, int, int, int)} and
386    * {@link #compress(byte[], int)} instead.
387    */
388   @Deprecated
compress(BufferedImage srcImage, byte[] dstBuf, int flags)389   public void compress(BufferedImage srcImage, byte[] dstBuf, int flags)
390                        throws Exception {
391     setSourceImage(srcImage, 0, 0, 0, 0);
392     compress(dstBuf, flags);
393   }
394 
395   /**
396    * @deprecated Use
397    * {@link #setSourceImage(BufferedImage, int, int, int, int)} and
398    * {@link #compress(int)} instead.
399    */
400   @Deprecated
compress(BufferedImage srcImage, int flags)401   public byte[] compress(BufferedImage srcImage, int flags) throws Exception {
402     setSourceImage(srcImage, 0, 0, 0, 0);
403     return compress(flags);
404   }
405 
406   /**
407    * Encode the uncompressed source image associated with this compressor
408    * instance into a YUV planar image and store it in the given
409    * <code>YUVImage</code> instance.   This method uses the accelerated color
410    * conversion routines in TurboJPEG's underlying codec but does not execute
411    * any of the other steps in the JPEG compression process.  Encoding
412    * CMYK source images to YUV is not supported.
413    *
414    * @param dstImage {@link YUVImage} instance that will receive the YUV planar
415    * image
416    *
417    * @param flags the bitwise OR of one or more of
418    * {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
419    */
encodeYUV(YUVImage dstImage, int flags)420   public void encodeYUV(YUVImage dstImage, int flags) throws Exception {
421     if (dstImage == null || flags < 0)
422       throw new Exception("Invalid argument in encodeYUV()");
423     if (srcBuf == null && srcBufInt == null)
424       throw new Exception(NO_ASSOC_ERROR);
425     if (srcYUVImage != null)
426       throw new Exception("Source image is not correct type");
427     if (subsamp < 0)
428       throw new Exception("Subsampling level not set");
429     if (srcWidth != dstImage.getWidth() || srcHeight != dstImage.getHeight())
430       throw new Exception("Destination image is the wrong size");
431 
432     if (srcBufInt != null) {
433       encodeYUV(srcBufInt, srcX, srcY, srcWidth, srcStride, srcHeight,
434                 srcPixelFormat, dstImage.getPlanes(), dstImage.getOffsets(),
435                 dstImage.getStrides(), dstImage.getSubsamp(), flags);
436     } else {
437       encodeYUV(srcBuf, srcX, srcY, srcWidth, srcPitch, srcHeight,
438                 srcPixelFormat, dstImage.getPlanes(), dstImage.getOffsets(),
439                 dstImage.getStrides(), dstImage.getSubsamp(), flags);
440     }
441     compressedSize = 0;
442   }
443 
444   /**
445    * @deprecated Use {@link #encodeYUV(YUVImage, int)} instead.
446    */
447   @Deprecated
encodeYUV(byte[] dstBuf, int flags)448   public void encodeYUV(byte[] dstBuf, int flags) throws Exception {
449     if(dstBuf == null)
450       throw new Exception("Invalid argument in encodeYUV()");
451     if (srcWidth < 1 || srcHeight < 1)
452       throw new Exception(NO_ASSOC_ERROR);
453     if (subsamp < 0)
454       throw new Exception("Subsampling level not set");
455     YUVImage yuvImage = new YUVImage(dstBuf, srcWidth, 4, srcHeight, subsamp);
456     encodeYUV(yuvImage, flags);
457   }
458 
459   /**
460    * Encode the uncompressed source image associated with this compressor
461    * instance into a unified YUV planar image buffer and return a
462    * <code>YUVImage</code> instance containing the encoded image.  This method
463    * uses the accelerated color conversion routines in TurboJPEG's underlying
464    * codec but does not execute any of the other steps in the JPEG compression
465    * process.  Encoding CMYK source images to YUV is not supported.
466    *
467    * @param pad the width of each line in each plane of the YUV image will be
468    * padded to the nearest multiple of this number of bytes (must be a power of
469    * 2.)
470    *
471    * @param flags the bitwise OR of one or more of
472    * {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
473    *
474    * @return a YUV planar image.
475    */
encodeYUV(int pad, int flags)476   public YUVImage encodeYUV(int pad, int flags) throws Exception {
477     if (srcWidth < 1 || srcHeight < 1)
478       throw new Exception(NO_ASSOC_ERROR);
479     if (subsamp < 0)
480       throw new Exception("Subsampling level not set");
481     if(pad < 1 || ((pad & (pad - 1)) != 0))
482       throw new Exception("Invalid argument in encodeYUV()");
483     YUVImage yuvImage = new YUVImage(srcWidth, pad, srcHeight, subsamp);
484     encodeYUV(yuvImage, flags);
485     return yuvImage;
486   }
487 
488   /**
489    * Encode the uncompressed source image associated with this compressor
490    * instance into separate Y, U (Cb), and V (Cr) image planes and return a
491    * <code>YUVImage</code> instance containing the encoded image planes.  This
492    * method uses the accelerated color conversion routines in TurboJPEG's
493    * underlying codec but does not execute any of the other steps in the JPEG
494    * compression process.  Encoding CMYK source images to YUV is not supported.
495    *
496    * @param strides an array of integers, each specifying the number of bytes
497    * per line in the corresponding plane of the output image.  Setting the
498    * stride for any plane to 0 is the same as setting it to the component width
499    * of the plane.  If <code>strides</code> is null, then the strides for all
500    * planes will be set to their respective component widths.  You can adjust
501    * the strides in order to add an arbitrary amount of line padding to each
502    * plane.
503    *
504    * @param flags the bitwise OR of one or more of
505    * {@link TJ#FLAG_BOTTOMUP TJ.FLAG_*}
506    *
507    * @return a YUV planar image.
508    */
encodeYUV(int[] strides, int flags)509   public YUVImage encodeYUV(int[] strides, int flags) throws Exception {
510     if (srcWidth < 1 || srcHeight < 1)
511       throw new Exception(NO_ASSOC_ERROR);
512     if (subsamp < 0)
513       throw new Exception("Subsampling level not set");
514     YUVImage yuvImage = new YUVImage(srcWidth, strides, srcHeight, subsamp);
515     encodeYUV(yuvImage, flags);
516     return yuvImage;
517   }
518 
519   /**
520    * @deprecated Use {@link #encodeYUV(int, int)} instead.
521    */
522   @Deprecated
encodeYUV(int flags)523   public byte[] encodeYUV(int flags) throws Exception {
524     if (srcWidth < 1 || srcHeight < 1)
525       throw new Exception(NO_ASSOC_ERROR);
526     if (subsamp < 0)
527       throw new Exception("Subsampling level not set");
528     YUVImage yuvImage = new YUVImage(srcWidth, 4, srcHeight, subsamp);
529     encodeYUV(yuvImage, flags);
530     return yuvImage.getBuf();
531   }
532 
533   /**
534    * @deprecated Use
535    * {@link #setSourceImage(BufferedImage, int, int, int, int)} and
536    * {@link #encodeYUV(byte[], int)} instead.
537    */
538   @Deprecated
encodeYUV(BufferedImage srcImage, byte[] dstBuf, int flags)539   public void encodeYUV(BufferedImage srcImage, byte[] dstBuf, int flags)
540     throws Exception {
541     setSourceImage(srcImage, 0, 0, 0, 0);
542     encodeYUV(dstBuf, flags);
543   }
544 
545   /**
546    * @deprecated Use
547    * {@link #setSourceImage(BufferedImage, int, int, int, int)} and
548    * {@link #encodeYUV(int, int)} instead.
549    */
550   @Deprecated
encodeYUV(BufferedImage srcImage, int flags)551   public byte[] encodeYUV(BufferedImage srcImage, int flags) throws Exception {
552     setSourceImage(srcImage, 0, 0, 0, 0);
553     return encodeYUV(flags);
554   }
555 
556   /**
557    * Returns the size of the image (in bytes) generated by the most recent
558    * compress operation.
559    *
560    * @return the size of the image (in bytes) generated by the most recent
561    * compress operation.
562    */
getCompressedSize()563   public int getCompressedSize() {
564     return compressedSize;
565   }
566 
567   /**
568    * Free the native structures associated with this compressor instance.
569    */
close()570   public void close() throws Exception {
571     if (handle != 0)
572       destroy();
573   }
574 
finalize()575   protected void finalize() throws Throwable {
576     try {
577       close();
578     } catch(Exception e) {
579     } finally {
580       super.finalize();
581     }
582   };
583 
init()584   private native void init() throws Exception;
585 
destroy()586   private native void destroy() throws Exception;
587 
588   // JPEG size in bytes is returned
compress(byte[] srcBuf, int width, int pitch, int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp, int jpegQual, int flags)589   private native int compress(byte[] srcBuf, int width, int pitch,
590     int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp, int jpegQual,
591     int flags) throws Exception; // deprecated
592 
compress(byte[] srcBuf, int x, int y, int width, int pitch, int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp, int jpegQual, int flags)593   private native int compress(byte[] srcBuf, int x, int y, int width,
594     int pitch, int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp,
595     int jpegQual, int flags) throws Exception;
596 
compress(int[] srcBuf, int width, int stride, int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp, int jpegQual, int flags)597   private native int compress(int[] srcBuf, int width, int stride,
598     int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp, int jpegQual,
599     int flags) throws Exception; // deprecated
600 
compress(int[] srcBuf, int x, int y, int width, int stride, int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp, int jpegQual, int flags)601   private native int compress(int[] srcBuf, int x, int y, int width,
602     int stride, int height, int pixelFormat, byte[] dstBuf, int jpegSubsamp,
603     int jpegQual, int flags) throws Exception;
604 
compressFromYUV(byte[][] srcPlanes, int[] srcOffsets, int width, int[] srcStrides, int height, int subsamp, byte[] dstBuf, int jpegQual, int flags)605   private native int compressFromYUV(byte[][] srcPlanes, int[] srcOffsets,
606     int width, int[] srcStrides, int height, int subsamp, byte[] dstBuf,
607     int jpegQual, int flags)
608     throws Exception;
609 
encodeYUV(byte[] srcBuf, int width, int pitch, int height, int pixelFormat, byte[] dstBuf, int subsamp, int flags)610   private native void encodeYUV(byte[] srcBuf, int width, int pitch,
611     int height, int pixelFormat, byte[] dstBuf, int subsamp, int flags)
612     throws Exception; // deprecated
613 
encodeYUV(byte[] srcBuf, int x, int y, int width, int pitch, int height, int pixelFormat, byte[][] dstPlanes, int[] dstOffsets, int[] dstStrides, int subsamp, int flags)614   private native void encodeYUV(byte[] srcBuf, int x, int y, int width,
615     int pitch, int height, int pixelFormat, byte[][] dstPlanes,
616     int[] dstOffsets, int[] dstStrides, int subsamp, int flags)
617     throws Exception;
618 
encodeYUV(int[] srcBuf, int width, int stride, int height, int pixelFormat, byte[] dstBuf, int subsamp, int flags)619   private native void encodeYUV(int[] srcBuf, int width, int stride,
620     int height, int pixelFormat, byte[] dstBuf, int subsamp, int flags)
621     throws Exception; // deprecated
622 
encodeYUV(int[] srcBuf, int x, int y, int width, int srcStride, int height, int pixelFormat, byte[][] dstPlanes, int[] dstOffsets, int[] dstStrides, int subsamp, int flags)623   private native void encodeYUV(int[] srcBuf, int x, int y, int width,
624     int srcStride, int height, int pixelFormat, byte[][] dstPlanes,
625     int[] dstOffsets, int[] dstStrides, int subsamp, int flags)
626     throws Exception;
627 
628   static {
TJLoader.load()629     TJLoader.load();
630   }
631 
632   private long handle = 0;
633   private byte[] srcBuf = null;
634   private int[] srcBufInt = null;
635   private int srcWidth = 0;
636   private int srcHeight = 0;
637   private int srcX = -1;
638   private int srcY = -1;
639   private int srcPitch = 0;
640   private int srcStride = 0;
641   private int srcPixelFormat = -1;
642   private YUVImage srcYUVImage = null;
643   private int subsamp = -1;
644   private int jpegQuality = -1;
645   private int compressedSize = 0;
646   private int yuvPad = 4;
647   private ByteOrder byteOrder = null;
648 };
649