OSDN Git Service

Handling the GcamService timeout due to application in paused state (camera-Haleakala)
[android-x86/packages-apps-Camera2.git] / jni / jpegutilnative.cpp
index 7012b97..b2afd94 100644 (file)
 
 #include "jpegutil.h"
 
+using namespace jpegutil;
+
 /**
+ * Compresses a YCbCr image to jpeg, applying a crop and rotation.
+ *
+ * The input is defined as a set of 3 planes of 8-bit samples, one plane for
+   each channel of Y, Cb, Cr.
+ * The Y plane is assumed to have the same width and height of the entire image.
+ * The Cb and Cr planes are assumed to be downsampled by a factor of 2, to have
+ * dimensions (floor(width / 2), floor(height / 2)).
+ * Each plane is specified by a direct java.nio.ByteBuffer, a pixel-stride, and
+ * a row-stride.  So, the sample at coordinate (x, y) can be retrieved from
+ * byteBuffer[x * pixel_stride + y * row_stride].
+ *
+ * The pre-compression transformation is applied as follows:
+ *  1. The image is cropped to the rectangle from (cropLeft, cropTop) to
+ *  (cropRight - 1, cropBottom - 1).  So, a cropping-rectangle of (0, 0) -
+ *  (width, height) is a no-op.
+ *  2. The rotation is applied counter-clockwise relative to the coordinate
+ *  space of the image, so a CCW rotation will appear CW when the image is
+ *  rendered in scanline order.  Only rotations which are multiples of
+ *  90-degrees are suppored, so the parameter 'rot90' specifies which multiple
+ *  of 90 to rotate the image.
+ *
  * @param env the JNI environment
+ * @param width the width of the image to compress
+ * @param height the height of the image to compress
  * @param yBuf the buffer containing the Y component of the image
  * @param yPStride the stride between adjacent pixels in the same row in yBuf
  * @param yRStride the stride between adjacent rows in yBuf
  * @param crBuf the buffer containing the Cr component of the image
  * @param crPStride the stride between adjacent pixels in the same row in crBuf
  * @param crRStride the stride between adjacent rows in crBuf
+ * @param outBuf a direct java.nio.ByteBuffer to hold the compressed jpeg.  This
+ * must have enough capacity to store the result, or an error code will be
+ * returned.
+ * @param outBufCapacity the capacity of outBuf
+ * @param quality the jpeg-quality (1-100) to use
+ * @param crop[Left|Top|Right|Bottom] the bounds of the image to crop to before
+ * rotation
+ * @param rot90 the multiple of 90 to rotate by
  */
 extern "C" JNIEXPORT jint JNICALL
-    Java_com_android_camera_util_JpegUtilNative_compressJpegFromYUV420pNative(
-        JNIEnv* env, jclass clazz, jint width, jint height, jobject yBuf,
-        jint yPStride, jint yRStride, jobject cbBuf, jint cbPStride,
-        jint cbRStride, jobject crBuf, jint crPStride, jint crRStride,
-        jobject outBuf, jint outBufCapacity, jint quality) {
+Java_com_android_camera_util_JpegUtilNative_compressJpegFromYUV420pNative(
+    JNIEnv* env, jclass clazz,
+    /** Input image dimensions */
+    jint width, jint height,
+    /** Y Plane */
+    jobject yBuf, jint yPStride, jint yRStride,
+    /** Cb Plane */
+    jobject cbBuf, jint cbPStride, jint cbRStride,
+    /** Cr Plane */
+    jobject crBuf, jint crPStride, jint crRStride,
+    /** Output */
+    jobject outBuf, jint outBufCapacity,
+    /** Jpeg compression parameters */
+    jint quality,
+    /** Crop */
+    jint cropLeft, jint cropTop, jint cropRight, jint cropBottom,
+    /** Rotation (multiple of 90).  For example, rot90 = 1 implies a 90 degree
+     * rotation. */
+    jint rot90) {
   jbyte* y = (jbyte*)env->GetDirectBufferAddress(yBuf);
   jbyte* cb = (jbyte*)env->GetDirectBufferAddress(cbBuf);
   jbyte* cr = (jbyte*)env->GetDirectBufferAddress(crBuf);
   jbyte* out = (jbyte*)env->GetDirectBufferAddress(outBuf);
 
-  jpegutil::Plane yP(width, height, width, height, (unsigned char*)y, yPStride,
-                     yRStride);
-  jpegutil::Plane cbP(width, height, width / 2, height / 2, (unsigned char*)cb,
-                      cbPStride, cbRStride);
-  jpegutil::Plane crP(width, height, width / 2, height / 2, (unsigned char*)cr,
-                      crPStride, crRStride);
-
-  auto flush = [](size_t numBytes) {
-    // do nothing
-  };
-
-  return jpegutil::compress(yP, cbP, crP, (unsigned char*)out, outBufCapacity,
-                            flush, quality);
+  return Compress(width, height,                                //
+                  (unsigned char*)y, yPStride, yRStride,        //
+                  (unsigned char*)cb, cbPStride, cbRStride,     //
+                  (unsigned char*)cr, crPStride, crRStride,     //
+                  (unsigned char*)out, (size_t)outBufCapacity,  //
+                  quality,                                      //
+                  cropLeft, cropTop, cropRight, cropBottom,     //
+                  rot90);
 }
 
 /**
@@ -67,18 +109,19 @@ extern "C" JNIEXPORT jint JNICALL
  * @param width the width of the output image
  * @param height the height of the output image
  * @param planeBuf the native ByteBuffer containing the image plane data
- * @param pStride the stride between adjacent pixels in the same row of planeBuf
+ * @param pStride the stride between adjacent pixels in the same row of
+ *planeBuf
  * @param rStride the stride between adjacent rows in planeBuf
  * @param rot90 the multiple of 90 degrees to rotate, one of {0, 1, 2, 3}.
  */
 extern "C" JNIEXPORT void JNICALL
-    Java_com_android_camera_util_JpegUtilNative_copyImagePlaneToBitmap(
-        JNIEnv* env, jclass clazz, jint width, jint height, jobject planeBuf,
-        jint pStride, jint rStride, jobject outBitmap, jint rot90) {
+Java_com_android_camera_util_JpegUtilNative_copyImagePlaneToBitmap(
+    JNIEnv* env, jclass clazz, jint width, jint height, jobject planeBuf,
+    jint pStride, jint rStride, jobject outBitmap, jint rot90) {
   jbyte* src = (jbyte*)env->GetDirectBufferAddress(planeBuf);
 
   char* dst = 0;
-  AndroidBitmap_lockPixels(env, outBitmap, (void**) &dst);
+  AndroidBitmap_lockPixels(env, outBitmap, (void**)&dst);
 
   if (rot90 == 0) {
     // No rotation