C++程序  |  168行  |  6.5 KB

/*
 * Copyright (C) 2014 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#include <jni.h>
#include <math.h>
#include <android/bitmap.h>

#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 cbBuf the buffer containing the Cb component of the image
 * @param cbPStride the stride between adjacent pixels in the same row in cbBuf
 * @param cbRStride the stride between adjacent rows in cbBuf
 * @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,
    /** 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);

  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);
}

/**
 * Copies the Image.Plane specified by planeBuf, pStride, and rStride to the
 * Bitmap.
 *
 * @param env the JNI environment
 * @param clazz the java class
 * @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 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) {
  jbyte* src = (jbyte*)env->GetDirectBufferAddress(planeBuf);

  char* dst = 0;
  AndroidBitmap_lockPixels(env, outBitmap, (void**)&dst);

  if (rot90 == 0) {
    // No rotation
    for (int y = 0; y < height; y++) {
      char* srcPtr = reinterpret_cast<char*>(&src[y * rStride]);
      char* dstPtr = &dst[y * width];
      for (int x = 0; x < width; x++) {
        *dstPtr = *srcPtr;
        srcPtr += pStride;
        dstPtr++;
      }
    }
  } else if (rot90 == 1) {
    // 90-degree rotation
    for (int y = 0; y < height; y++) {
      for (int x = 0; x < width; x++) {
        int srcX = height - 1 - y;
        int srcY = x;
        dst[y * width + x] = src[srcX * pStride + rStride * srcY];
      }
    }
  } else if (rot90 == 2) {
    // 180-degree rotation
    for (int y = 0; y < height; y++) {
      for (int x = 0; x < width; x++) {
        int srcX = width - 1 - x;
        int srcY = height - 1 - y;
        dst[y * width + x] = src[srcX * pStride + rStride * srcY];
      }
    }
  } else if (rot90 == 3) {
    // 270-degree rotation
    for (int y = 0; y < height; y++) {
      for (int x = 0; x < width; x++) {
        int srcX = y;
        int srcY = width - 1 - x;
        dst[y * width + x] = src[srcX * pStride + rStride * srcY];
      }
    }
  }

  AndroidBitmap_unlockPixels(env, outBitmap);
}